1 R実行環境の整備

1.1 Posit Cloudへのサインアップ

Rを使って統計分析を行う際には、本来はRRStudioという2つのアプリを自分のPCにインストールする必要があるが、インターネットに接続できている場合には、Posit Cloudというオンラインサービスを利用することによって、これらのアプリを自分のPCにインストールすることなくWebブラウザ上でRStudioを利用することができる。

今回はPosit Cloudを利用することを前提に説明していく。

Posit Cloudにアクセス。 下記の通りのページが表示される。

Posit Cloud
Posit Cloud

「Get Started」をクリック

Freeプランの「Learn more」をクリック

「Sign Up」をクリック。下記のページが表示される。

Sign Up
Sign Up

「Sign Up with Google」を選択して、大学のGmailアドレスで登録。

「Posit Cloudにログイン」の画面で「次へ」を選択すると、サインアップが完了し、以下のような画面が表示される。

WorkspaceTop
WorkspaceTop

なお、Posit Cloudは英語のサイトだが、GoogleChromeの「翻訳」で「日本語」にすればある程度は日本語表記がされる。(あくまである程度であって、Rを操作する画面までは残念ながら日本語化されない)

1.2 Procjectの作成

右上にある「New Project」をクリック

「New RStduio Project」をクリック。「Deploying Project」とプログレスバーがしばらく表示されたあと、以下のような画面が表示される。

RStduio
RStduio

この画面がR(Rstduio)の基本操作画面になる。

1.3 Project名の設定

続いて、プロジェクトに名前を付ける。 画面トップにある「Untitled Project」と書かれてある箇所をクリックして、名前を書き入れよう。

ProjectName
ProjectName

ただし、注意しないといけない点として、Project名は半角文字しか受け付けてくれない(アルファベットや数字、アンダーバーなど)。日本語を書き入れても、最初は表示されてもリロードされたときに「?」という表記に変わってしまう。

日本語名入力
日本語名入力

ページをリロード(再読み込み)すると、以下のようになる。

日本語名
日本語名

2 RとRStudioの基本操作

2.1 基本的な計算の実行

「Console」と書かれた以下の画面に数式を入力することによって、Rに計算をさせることができる。

Console
Console

実際に各自で好きな数で四則演算をさせてみよう。以下は一例。

5+1
## [1] 6
5-7
## [1] -2
6*4
## [1] 24
9/3
## [1] 3

さらには、カッコを使ったり、累乗演算をさせたり、剰余演算をさせたりもできる。

# カッコつき
5*3*(11-23)
## [1] -180
# カッコを重ねることもできる(中カッコ[]や大カッコ{}は用いない)
5+3*(11-23*(4+3))
## [1] -445
# 累乗演算
2^3
## [1] 8
# 剰余演算
15%%6
## [1] 3

このように、+-*/()^%%などの演算記号のことを演算子(Operator)と呼ぶ。演算子にはこれら以外にもいろいろある。以下にRで使われる代表的な演算子を示す。

演算子 説明
+ 加算 x + y
- 減算 x - y
* 乗算 x * y
/ 除算 x / y
^ 累乗 x ^ y
%% 剰余 x %% y
%/% 整数除算 x %/% y
== 等しい x == y
!= 等しくない x != y
< より小さい x < y
> より大きい x > y
<= 以下 x <= y
>= 以上 x >= y
& 論理積 (AND) x & y
| 論理和 (OR) x | y
! 否定 (NOT) !x
%in% ベクトルの要素が含まれるか x %in% y
: シーケンスの生成 1:10
<- 代入 x <- 10
= 代入 (引数で使用) func(x = 10)

平方根の場合には、sqrt()という関数を使う。

sqrt(16)
## [1] 4

ちなみに、出力結果の前にある[1]という表記は、出力結果が長さ1のベクトル(Vector)の1つ目の要素であることを示している。このベクトルについては後で改めて説明する。

なお、このようにconsoleに1行1行数式や関数(まとめてコマンド)を入力していく形をCommand Line Interface (CLI) と呼ぶ。CLIでは、操作できるのはあくまで現在カーソルが置かれている行だけである。過去の行はログとして見えているだけであり、ログを消したり、書き換えたりはできない。

もし誤ったコマンドを入力しているのに気づかずに実行してしまった場合には、もう一度正しいコマンドを改めて入力すればよい。

2.2 オブジェクト

Rではオブジェクトと呼ばれる箱に計算結果の数値や読み込んだデータ(後で説明する)を格納することができる。オブジェクトに格納しておくことと、後からそれらの値やデータを再度利用したくなったときに、いちいち再度計算式を入力したり、データの読み込みをすることなく、オブジェクトを呼び出すだけで、それらの値やデータを呼び出すことができる。

以下の例では5-3の結果をResという名前のオブジェクトに格納し、さらにResというオブジェクト名を使ってその中身(計算結果)を呼び出している。

Res <- 5-3
Res
## [1] 2

2.2.1 print()関数

オブジェクトの内容は、上記のようにオブジェクト名を記載するだけでも出力されるが、print()関数使って出力させることもできる。なお、print()関数はオブジェクトだけでなく数字や文字列も表示させることが出来る。

## print()関数を使って表記させることも可能
print(Res)
## [1] 2
print(1)
## [1] 1
print("Hello World.")
## [1] "Hello World."

2.2.2 オブジェクトを使った演算

オブジェクトに格納された値は、そのままオブジェクト名で計算させることもできる。

8+Res
## [1] 10

2.2.3 オブジェクトの名前設定

オブジェクト名はどのような名前でも構わない。ただし、半角スペースや演算子に用いられている記号、アットマーク記号@などは利用することができない。また、数字から始まる名前を付けることも不可である(名前の途中に数字が入っている分には問題ない)。 ピリオド.やアンダーバー_は利用できる。一般的には、複数の単語を連結させて用いる場合には_で区切る。.は階層性を持ったデータを区別する使われる。以下はそれらの例。

my_res <- 9-4
my_res.power <- my_res ^3

なお、間違えた名前で呼び出すと、もしその名前のオブジェクトが他にないのであれば、以下のようなエラーが帰ってくる

RES
## Error in eval(expr, envir, enclos):  オブジェクト 'RES' がありません

「オブジェクトが見つかりません」ということで、このエラーが帰ってきたら、オブジェクト名が間違えているかを確認してほしい。

ただし、オブジェクト名を間違えていた時に、間違えた名前と同一の名前のオブジェクトが別に存在していた場合には、このようなエラーは帰って来ずに、そのまま間違えたオブジェクトを使って続く処理が実行されていくので、その場合には自分で間違いに気づきにくくなる。RStduioではすでに使われているオブジェクト名であれば、最初の数文字を入力するとリスト表示してくれる入力支援機能があるので、オブジェクトが増えてきたときにはそれを是非利用して入力するとよい。

2.3 文字列

これまで数値ばかりを扱ってきたが、Rでは数値だけでなく文字列を扱うこともできる。

nchar("Hidenori Fujino")
## [1] 15

nchar()は文字列の長さを求める関数である。ほかにも検索をしたり、置換したり、分割したり、結合したりといったことができる関数が用意されている。

また、オブジェクトに文字列を格納することもできる。

myName <-"Hidenori Fujino"
myName
## [1] "Hidenori Fujino"
nchar(myName)
## [1] 15

2.4

2.4.1 データの型

オブジェクトにどのようなデータが収められているかによって、そのオブジェクトには「型」と呼ばれる属性が付与される。数値が収められていれば、そのオブジェクトは数値型(Numeric)となるし、文字列が収められていれば文字列型(Character)となる。ほかにも論理型(Logical)や日付型(Date)などいろいろな型がある。

厳密には、オブジェクトに入れていない数値や文字列そのものも、それぞれ数値型や文字列型のデータである。つまり型とはデータそのものに付随する属性であり、オブジェクトの型=その中見のデータの型である。

基本的に型が異なる者同士を計算させることはできない。

a <- 5
b <- "2"
a-b
## Error in a - b:  二項演算子の引数が数値ではありません

この例では、bには2ではなく"2"という文字列が格納されている。このため数値‐文字列という形で方の異なるもの同士を計算しようとしたので、エラーとなった。

ただし、関数の場合には、数値でもそのまま文字列をとして扱ってくれるものもある。

a <- 12345
b <- "12345"
nchar(a)
## [1] 5
nchar(b)
## [1] 5

2.4.2 型の把握

あるオブジェクトがどの型なのかを知るためには、class()関数を用いる。

a <- 12345
b <- "12345"
class(a)
## [1] "numeric"
class(b)
## [1] "character"

2.4.3 型の変換

たとえば文字列として与えられた数値を数値型に変換して処理したり、逆に数値型として与えられた数値を文字列型に変換したい、といったケースはよくある。 その場合、as.numericas.characterといった型変換関数を用いる。

例えば、以下は文字列型の数値を数値型に変換している例である。

a <- 12345
b <- "12345"
a+b # これはエラー
## Error in a + b:  二項演算子の引数が数値ではありません
a + as.numeric(b) 
## [1] 24690

このほかにも、型変換関数はたくさんある。as.まで打つと様々な型変換関数がリスト表記される。データ分析でよく使用するものとしてはas.factor()(データを要因型(参考)に変換する)やas.data.frame()(データをデータフレーム型に変換する)などがある。それぞれいずれ出てくるので、頭の片隅に置いておいてほしい。

2.5 ベクトル

2.5.1 ベクトルの作成

Rではこれまでの例のように単独の数値や文字列を扱うケースは極めて稀である。通常は複数の数値や文字列を1セットにして様々な処理を行っていく。そうした1セットにしたデータセットの最もシンプルなものがベクトル(1次元配列)である。

例えば、バスケチームのStartingメンバー5名でフリースロー10本勝負をしたときの成績として以下のような成績が出たとする。

名前 点数
赤木 7
桜木 4
三井 10
宮城 7
流川 9

この表には名前のデータセットと得点のデータセットがある。これらがそれぞれベクトルとなる。Rでそれぞれのベクトルを作成するには、c()という関数を使う。cはconcatenate(連結する)の頭文字である。

Starting <- c("赤木","桜木","三井","宮城","流川")
Score<- c(7,  4, 10, 7, 9)
print(Starting)
## [1] "赤木" "桜木" "三井" "宮城" "流川"
print(Score)
## [1]  7  4 10  7  9

c()関数を使うと、複数のベクトルを1つのベクトルにまとめることもできる。

Reserve <-c("小暮","潮崎","安田","石井","佐々岡","桑田")
Full <- c(Starting, Reserve)
print(Full)
##  [1] "赤木"   "桜木"   "三井"   "宮城"   "流川"   "小暮"   "潮崎"   "安田"  
##  [9] "石井"   "佐々岡" "桑田"

データ上あまり意味はないが、文字列型のデータが収められたベクトルと数値型のデータが収められたベクトルをc()で連結することもできる。この場合、数値型のデータが文字列型に自動的に型変換される。

hoge <- c(Starting, Score)
print(hoge) #Scoreのデータが文字列型として出力される。
##  [1] "赤木" "桜木" "三井" "宮城" "流川" "7"    "4"    "10"   "7"    "9"

名前付きベクトル

先ほど作成したScoreは単に数値だけが並ぶベクトルであるが、以下のように各数値に「名前」を付けることができる。このようなベクトルを名前付きベクトルと呼ぶ。 なお、通常の文字列の扱いでは""で括らなければならないが、名前に対しては""で括らずにそのまま書いてもよいし、""で括ってもよい。

NamedScore <-  c(赤木=7,  桜木=4, 三井=10, 宮城=7, 流川=9)
print(NamedScore)
## 赤木 桜木 三井 宮城 流川 
##    7    4   10    7    9
NamedScore <-  c("赤木"=7,  "桜木"=4, "三井"=10, "宮城"=7, "流川"=9)
print(NamedScore)
## 赤木 桜木 三井 宮城 流川 
##    7    4   10    7    9

名前を付けておくとベクトル内の個々のデータにアクセスする際に、インデックス番号だけでなく、名前でアクセスすることもできる。詳しくは個々のデータへのアクセスへ。

2.5.2 ベクトルの型

ベクトルオブジェクトの型はそのベクトルが内部に持つデータの型によって決まる。

class(Starting)
## [1] "character"
class(Score)
## [1] "numeric"

2.5.3 個々のデータへのアクセス

ベクトルオブジェクトの内の個別のデータにアクセスする(取り出す)には[]を使って取り出したいデータのインデックス番号を指定する。

Full[1]
## [1] "赤木"
Score[3]
## [1] 10

複数のデータを同時に取り出すには[]の中へ、c()を使って複数の番号を指定する。

Starting[c(3,5)]
## [1] "三井" "流川"
Score[c(3,5)]
## [1] 10  9

例えば2番目から5番目のデータを取り出したいという場合には、いちいちc(2,3,4,5)と打つのは面倒である。 そういう場合にはシーケンス演算子:を用いて以下のように書くことができる。

Starting[c(2,3,4,5)] #めんどくさい
## [1] "桜木" "三井" "宮城" "流川"
Starting[c(2:5)]
## [1] "桜木" "三井" "宮城" "流川"

通常の指定とシーケンス演算子での指定は併せて用いることもできる。

Starting[c(1,3,4,5)] #めんどくさい
## [1] "赤木" "三井" "宮城" "流川"
Starting[c(1,3:5)]
## [1] "赤木" "三井" "宮城" "流川"

名前付きベクトルの場合には、インデックス番号の代わりに名前を指定してもよい。

NamedScore[5]
## 流川 
##    9
NamedScore["流川"]
## 流川 
##    9
NamedScore[c(1,3,4)]
## 赤木 三井 宮城 
##    7   10    7
NamedScore[c("赤木","三井","宮城")]
## 赤木 三井 宮城 
##    7   10    7

名前付きベクトルの場合には上記の通りに名前とベクトル値がセットになって返されてくる。 あくまで数値だけが欲しいという場合には、unname()関数を用いる。あるいは、要素が1つだけの場合には[[]]というように角カッコを重ねることでもよい。ただし、[[]]が使えるのは要素が1つの時だけであり、複数の要素に対して用いるとエラーとなる。

unname(NamedScore[[5]])
unname(NamedScore[c(1,3,4)]
unname(NamedScore[c("赤木","三井","宮城")])
NamedScore[["流川"]]
NamedScore[[c(1,3,4)]] #これはエラー
NamedScore[[c("赤木","三井","宮城")]] #これもエラー
## Error: <text>:3:1:  想定外のシンボルです 
## 2: unname(NamedScore[c(1,3,4)]
## 3: unname
##    ^

2.5.4 個々のデータの削除

インデックス番号に-記号を付けると、その番号のデータが除かれたものが返ってくる。

Full[-3] #3番目の三井が除かれる
##  [1] "赤木"   "桜木"   "宮城"   "流川"   "小暮"   "潮崎"   "安田"   "石井"  
##  [9] "佐々岡" "桑田"
Full[c(-4,-6)] # 4番目の宮城と6番目の木暮が除かれる
## [1] "赤木"   "桜木"   "三井"   "流川"   "潮崎"   "安田"   "石井"   "佐々岡"
## [9] "桑田"
Full[c(4,6,-7)] #これはエラー。取り出しと除外を同時できない。
## Error in Full[c(4, 6, -7)]:  負の添字と混在できるのは 0 という添字だけです

2.5.5 個々のデータの書き換え

ベクトル内のデータへのアクセスと同じ要領で書き換えたいデータを指定し、代入演算子を使って値を与えればよい

Starting[1] <- "Akagi"
Starting
## [1] "Akagi" "桜木"  "三井"  "宮城"  "流川"

同時に複数のものを書き換えることもできる。

Starting[c(1,2)] <- c("Akagi", "Sakuragi")
Starting
## [1] "Akagi"    "Sakuragi" "三井"     "宮城"     "流川"

2.5.6 ベクトルと数値の演算

ベクトルに対して数値を使って演算をすると、ベクトル内のすべてのデータに対してその数値の演算が行われる。

Score + 10
## [1] 17 14 20 17 19
Score - 3
## [1] 4 1 7 4 6
Score * 2
## [1] 14  8 20 14 18
Score / 4
## [1] 1.75 1.00 2.50 1.75 2.25
Score ^ 2
## [1]  49  16 100  49  81

また名前付きベクトルの場合には、名前そのものは演算には影響せず、データだけに演算がされる。演算結果は名前付きベクトルが返ってくる。

NamedScore + 10
## 赤木 桜木 三井 宮城 流川 
##   17   14   20   17   19
NamedScore - 3
## 赤木 桜木 三井 宮城 流川 
##    4    1    7    4    6
NamedScore * 2
## 赤木 桜木 三井 宮城 流川 
##   14    8   20   14   18
NamedScore / 4
## 赤木 桜木 三井 宮城 流川 
## 1.75 1.00 2.50 1.75 2.25
NamedScore ^ 2
## 赤木 桜木 三井 宮城 流川 
##   49   16  100   49   81

2.5.7 ベクトル同士の演算

同じ長さのベクトルであればベクトル同士の掛け算ができる。

hoge <- c(3,5,0,2,1)
Score + hoge
## [1] 10  9 10  9 10
Score * hoge
## [1] 21 20  0 14  9
Score / hoge #3つ目の分母が0になるので、Infinity、つまり無限大を表すInfが表示される
## [1] 2.333333 0.800000      Inf 3.500000 9.000000

名前付きベクトルの場合も同様。

NamedScore + hoge
## 赤木 桜木 三井 宮城 流川 
##   10    9   10    9   10
NamedScore * hoge
## 赤木 桜木 三井 宮城 流川 
##   21   20    0   14    9
NamedScore / hoge 
##     赤木     桜木     三井     宮城     流川 
## 2.333333 0.800000      Inf 3.500000 9.000000

さらに名前付きベクトル同士を演算させた場合。

namedhoge<- c(A=3,B=5,C=0,D=2,E=1)
NamedScore + namedhoge
## 赤木 桜木 三井 宮城 流川 
##   10    9   10    9   10
namedhoge + NamedScore
##  A  B  C  D  E 
## 10  9 10  9 10
hoge + NamedScore + namedhoge
## 赤木 桜木 三井 宮城 流川 
##   13   14   10   11   11

このように名前付きベクトル同士を演算させた場合には、式の中で最初に出てくる名前付きベクトルの名前だけが出力に反映される。

演算するベクトルの長さが異なっているとWarningが出力される。興味深いことに、Errorではなく、あくまでWarningであって計算結果は出力される。長さが短い方が長い方に合わせて循環していく。また、長い方のベクトルの長さが短い方のベクトルの長さの整数倍の時には、Warningも表示されない。

hoge2 <- c(3,5,0,2,1,4,3)
Score + hoge2 # 長さが合ってないのでWarningが出てくる
## Warning in Score + hoge2:
## 長いオブジェクトの長さが短いオブジェクトの長さの倍数になっていません
## [1] 10  9 10  9 10 11  7
hoge3 <- c(3,5,0,2,1,4,3,3,4,4) # hoge3の長さがScoreの2倍になっているのでWarningはでない
Score + hoge3 
##  [1] 10  9 10  9 10 11  7 13 11 13

2.5.8 ベクトルへの要素の追加

ベクトルに要素を追加したい場合には、追加されるベクトルと追加したい要素やベクトルをc()で結び付けて、元のベクトルオブジェクトに格納すればよい。もちろん、元のベクトルではなく別のベクトルオブジェクトとして格納することもできる

print(Starting)
## [1] "Akagi"    "Sakuragi" "三井"     "宮城"     "流川"
Starting <- c(Starting, "小暮")
print(Starting)
## [1] "Akagi"    "Sakuragi" "三井"     "宮城"     "流川"     "小暮"
Reserve <- c("潮崎","安田","角田","石井","佐々岡","桑田")
Full <- c(Starting, Reserve)
print(Full)
##  [1] "Akagi"    "Sakuragi" "三井"     "宮城"     "流川"     "小暮"    
##  [7] "潮崎"     "安田"     "角田"     "石井"     "佐々岡"   "桑田"

2.5.9 ベクトルからの要素の削除

ベクトルから特定の要素を削除したい場合には、先に説明した個々のデータへのアクセスの方法に従って、残したい要素だけを取り出し、それをもとのベクトルオブジェクトに格納すればよい。

print(Starting) # 小暮が入っている
## [1] "Akagi"    "Sakuragi" "三井"     "宮城"     "流川"     "小暮"
Starting <- Starting[c(1:5)]
print(Starting) # 小暮が除外された
## [1] "Akagi"    "Sakuragi" "三井"     "宮城"     "流川"

2.6 データフレーム

ベクトルはあくまで1次元配列であるが、実際のデータセットは行と列を持った2次元配列(要するに表)になっている。

名前 背番号 点数
赤木 4 7
桜木 10 4
三井 14 10
宮城 7 7
流川 11 9

このような表としてのデータを一まとめにして扱うのがデータフレームという型のオブジェクトである。

2.6.1 データフレームの作成

データフレームを作るには以下のようにdata.frame()関数を用いる。

Shohoku <- data.frame(
  name = c("赤木","桜木","三井","宮城","流川"),
  number=c(4,10,13,7,11),
  score= c(7,4,10,7,9)
)
Shohoku
##   name number score
## 1 赤木      4     7
## 2 桜木     10     4
## 3 三井     13    10
## 4 宮城      7     7
## 5 流川     11     9

data.frame()関数の中でnamenumberscoreの3つのベクトルが作られている。つまりデータフレームとは同じ長さのベクトルが複数まとまったものと捉えることができる。

上の例では分かりやすさのためdata.frame()の中で改行(Enter)を入れているが、改行は入れても入れなくても良い。console上で()""''の中で改行を入れた場合、行頭が>から+にかわり、直近の("'の範囲(スコープと呼ぶ)閉じられずに続いていることを示す。スコープ閉じられていない限りはどれだけEnterを押してもコマンドは実行されない。適切にこれらが閉じられた場合には、閉じられた行でEnterキーを押すとことで一連の入力が一つながりのコマンドとして実行される。

なお、データフレームにおいては、行のことをobs(observation:観測の略)、列のことをvarialbles(変数の意味)と呼ぶ。これは行方向(縦方向)は観察・測定されたサンプルで分かれており、列方向(横方向)は測定項目・測定変数を表している、ということである。

2.6.2 列(ベクトル)へのアクセス

データフレームの各列のベクトルにアクセスするには、$もしくは[]を使う。

Shohoku$name # ベクトルとして出力される
## [1] "赤木" "桜木" "三井" "宮城" "流川"
Shohoku["name"] # nameだけを含んだデータフレームとして出力される
##   name
## 1 赤木
## 2 桜木
## 3 三井
## 4 宮城
## 5 流川
Shohoku[1] #[]を使う場合には列の番号を与えても良い
##   name
## 1 赤木
## 2 桜木
## 3 三井
## 4 宮城
## 5 流川

両者の出力結果は一見似ているが、$の方はベクトルそのものが出力されるのに対して、[]の方は指定したベクトルだけを含んだデータフレームが出力される。 それぞれの違いを確認するには、出力の型を確認すればよい。

class(Shohoku$name) # ベクトルなので、文字列型となる
## [1] "character"
class(Shohoku["name"])#データフレームなのでdata.frame型となる
## [1] "data.frame"

複数の列(ベクトル)を同時に取り出す場合にはc()関数を使って複数の列を指定してやる。

Shohoku[c("name","score")]
##   name score
## 1 赤木     7
## 2 桜木     4
## 3 三井    10
## 4 宮城     7
## 5 流川     9
Shohoku[c(1,3)] #名前ではなく列番号で指定した場合
##   name score
## 1 赤木     7
## 2 桜木     4
## 3 三井    10
## 4 宮城     7
## 5 流川     9

この場合の出力結果はdata.frame型となる。

2.6.3 列(ベクトル)の書き換え

列にアクセスするのと同じ要領で列を指定し、代入演算子によって書き換えたいベクトルを与えてやる。以下では$で指定する方法で名前をローマ字に書き換えた後、[]で指定する方法でローマ字になった名前を漢字に戻している。

Shohoku$name <- c("Akagi","Sakuragi","Mitsui","Miyagi","Rukawa")
Shohoku
##       name number score
## 1    Akagi      4     7
## 2 Sakuragi     10     4
## 3   Mitsui     13    10
## 4   Miyagi      7     7
## 5   Rukawa     11     9
Shohoku[1] <- c("赤木","桜木","三井","宮城","流川")
Shohoku
##   name number score
## 1 赤木      4     7
## 2 桜木     10     4
## 3 三井     13    10
## 4 宮城      7     7
## 5 流川     11     9

注意しないといけない点として、あくまで元の列ベクトルに含まれるデータ数(要するに行数)と同じデータ数のベクトルを与えなければならない。異なるデータ数のベクトルを与えた場合、Errorが返される。

Shohoku$name <- c("Akagi","Sakuragi","Mitsui","Miyagi") #データが1つ足りない
## Error in `$<-.data.frame`(`*tmp*`, name, value = c("Akagi", "Sakuragi", :  置換は 4 列ですが、データは 5 列です
Shohoku[1] <- c("Akagi","Sakuragi","Mitsui","Miyagi") #データが1つ足りない
## Error in `[<-.data.frame`(`*tmp*`, 1, value = c("Akagi", "Sakuragi", "Mitsui", :  置換は 4 列ですが、データは 5 列です

与えるベクトルデータが数値の場合には、演算を伴ったものでもよい。

Shohoku$score <- Shohoku$score + 5 # 5を足した
Shohoku
##   name number score
## 1 赤木      4    12
## 2 桜木     10     9
## 3 三井     13    15
## 4 宮城      7    12
## 5 流川     11    14
Shohoku[3] <- Shohoku$score - 5 # 5を引いた
Shohoku
##   name number score
## 1 赤木      4     7
## 2 桜木     10     4
## 3 三井     13    10
## 4 宮城      7     7
## 5 流川     11     9
Shohoku$score <- Shohoku$score * 3
Shohoku
##   name number score
## 1 赤木      4    21
## 2 桜木     10    12
## 3 三井     13    30
## 4 宮城      7    21
## 5 流川     11    27
Shohoku[3] <- Shohoku$score / 3
Shohoku
##   name number score
## 1 赤木      4     7
## 2 桜木     10     4
## 3 三井     13    10
## 4 宮城      7     7
## 5 流川     11     9

2.6.4 個々のデータへのアクセス

方法としては$を使ってベクトルを取り出した上で[]で取り出したい値の番号を指定する方法(これは要するにベクトルでの個々のデータへのアクセスと同じ)と、[]を使って行番号と列番号を指定する方法とがある。

Shohoku$name[1]
## [1] "赤木"
Shohoku[1,1]
## [1] "赤木"

なお[行, 列]である。

Shohoku[1,1]
## [1] "赤木"
Shohoku[1,2]
## [1] 4
Shohoku[1,3]
## [1] 7

列番号は列名(ベクトル名)で指定しても良い。

Shohoku[2,"name"]
## [1] "桜木"
Shohoku[2,"number"]
## [1] 10
Shohoku[2,"score"]
## [1] 4

$にせよ[]にせよ、複数の要素を同時に取り出すことができる。

Shohoku$name[c(1,3,5)]  # これは単純にベクトルとして取り出した後に[]でベクトル要素をしている
## [1] "赤木" "三井" "流川"
Shohoku[c(1,3,5),1] #これはデータフレームから行番号と列番号に合致するものを直接とりだしている
## [1] "赤木" "三井" "流川"
Shohoku[c(1,3,5),"name"] #列番号をベクトル名で指定した
## [1] "赤木" "三井" "流川"

[]を使う場合で列を1つしか指定しなかった場合には上記のようにベクトルとしてデータが返される。一方、以下のように列を複数指定した場合には、データフレームとしてデータが返される。

Shohoku[c(1,3,5),c(1,3)] #複数の列から複数の行番号のデータを同時に取り出そうとしている。
##   name score
## 1 赤木     7
## 3 三井    10
## 5 流川     9
Shohoku[c(1,3,5),c("name","score")]#列指定を名前にした
##   name score
## 1 赤木     7
## 3 三井    10
## 5 流川     9

2.6.5 個々のデータの書き換え

ベクトルの個々のデータの書き換えと同様に、それぞれの箇所にアクセスして代入演算子で書き換えてやる。以下では前と同様に$で指定する方法で名前をローマ字に替えた後、[]で指定する方法で名前を漢字に戻している

Shohoku$name[1] <- "Akagi"
Shohoku
##    name number score
## 1 Akagi      4     7
## 2  桜木     10     4
## 3  三井     13    10
## 4  宮城      7     7
## 5  流川     11     9
Shohoku[1,1] <-"赤木"
Shohoku
##   name number score
## 1 赤木      4     7
## 2 桜木     10     4
## 3 三井     13    10
## 4 宮城      7     7
## 5 流川     11     9

複数行を同時に書き換える場合には、ベクトルデータを与えてやる。

Shohoku$name[c(1,3,5)]<-c("Akagi","Mitsui","Rukawa")
Shohoku
##     name number score
## 1  Akagi      4     7
## 2   桜木     10     4
## 3 Mitsui     13    10
## 4   宮城      7     7
## 5 Rukawa     11     9
Shohoku[c(1,3,5),1] <-c("赤木","三井","流川")
Shohoku
##   name number score
## 1 赤木      4     7
## 2 桜木     10     4
## 3 三井     13    10
## 4 宮城      7     7
## 5 流川     11     9

複数行、複数列を同時に書き換えたい場合には、以下のようにdata.frame()関数を使ってデータを与える。

Shohoku[c(1,3,5),c(1,3)] <-data.frame(c("Akagi","Mitsui","Rukawa"),c(1,1,1))
Shohoku
##     name number score
## 1  Akagi      4     1
## 2   桜木     10     4
## 3 Mitsui     13     1
## 4   宮城      7     7
## 5 Rukawa     11     1
Shohoku[c(1,3,5),c("name","score")] <-data.frame(c("赤木","三井","流川"),c(7,10,9))
Shohoku
##   name number score
## 1 赤木      4     7
## 2 桜木     10     4
## 3 三井     13    10
## 4 宮城      7     7
## 5 流川     11     9

2.6.6 列(ベクトル)の追加

既にあるデータフレームに新しい列(ベクトル)を追加したい場合には以下のように$[]を用いて、新しい列名(ベクトル名)を指定して、その列に入れるデータをベクトルで与えるだけでよい。追加された列はデータフレームの末尾に入れられる。

Shohoku$Initial <- c("T.A.","H.S.","H.M.","R.M.","K.R.")
Shohoku
##   name number score Initial
## 1 赤木      4     7    T.A.
## 2 桜木     10     4    H.S.
## 3 三井     13    10    H.M.
## 4 宮城      7     7    R.M.
## 5 流川     11     9    K.R.
Shohoku["Yomi"] <- c("あかぎたけのり","さくらぎはなみち","みついひさし","みやぎりょうた","るかわかえで")
Shohoku
##   name number score Initial             Yomi
## 1 赤木      4     7    T.A.   あかぎたけのり
## 2 桜木     10     4    H.S. さくらぎはなみち
## 3 三井     13    10    H.M.     みついひさし
## 4 宮城      7     7    R.M.   みやぎりょうた
## 5 流川     11     9    K.R.     るかわかえで

以下の例のように、[]を使って複数の列を同時に追加することもできる。ただし、実用の場面では記述している処理の分かりやすさの観点から、1つ1つ追加していく方が良い。

Rの場合、それほど処理速度に敏感になる必要がないので、人から見た分かりやすさを重視した記述を心掛けるようにしよう。一方で、PythonやJavascript、あるいはC言語系などのプログラム言語を使って、動画像処理などリアルタイム性や高速な処理が求められるプログラムを記述する場合には、分かりやすさを犠牲にして、コンピュータ内部での計算処理回数の少ない書き方をすることもある。

Shohoku[c("UnderName","Roman")] <-data.frame(
  c("剛憲","花道","寿","亮太","楓"),
  c("Akagi","Sakuragi","Mitsui","Miyagi","Rukawa")
)
Shohoku
##   name number score Initial             Yomi UnderName    Roman
## 1 赤木      4     7    T.A.   あかぎたけのり      剛憲    Akagi
## 2 桜木     10     4    H.S. さくらぎはなみち      花道 Sakuragi
## 3 三井     13    10    H.M.     みついひさし        寿   Mitsui
## 4 宮城      7     7    R.M.   みやぎりょうた      亮太   Miyagi
## 5 流川     11     9    K.R.     るかわかえで        楓   Rukawa

2.6.7 列の削除

列を削除する場合には、削除したい列にアクセスしてNULL(ヌル)を代入すればよい。

Shohoku$Yomi <- NULL
Shohoku
##   name number score Initial UnderName    Roman
## 1 赤木      4     7    T.A.      剛憲    Akagi
## 2 桜木     10     4    H.S.      花道 Sakuragi
## 3 三井     13    10    H.M.        寿   Mitsui
## 4 宮城      7     7    R.M.      亮太   Miyagi
## 5 流川     11     9    K.R.        楓   Rukawa
Shohoku["Initial"] <- NULL
Shohoku
##   name number score UnderName    Roman
## 1 赤木      4     7      剛憲    Akagi
## 2 桜木     10     4      花道 Sakuragi
## 3 三井     13    10        寿   Mitsui
## 4 宮城      7     7      亮太   Miyagi
## 5 流川     11     9        楓   Rukawa
Shohoku[c("UnderName","Roman")] <- data.frame(NULL,NULL)
Shohoku
##   name number score
## 1 赤木      4     7
## 2 桜木     10     4
## 3 三井     13    10
## 4 宮城      7     7
## 5 流川     11     9

2.6.8 行の追加

データフレームへの行の追加にはrbind()関数を使う。以下では同じ列名を持ったkogureというデータフレームを作成し、Shouhokuに追加している。

kogure <- data.frame(name="木暮", number=5, score=6)
Shohoku <- rbind(Shohoku , kogure)
Shohoku
##   name number score
## 1 赤木      4     7
## 2 桜木     10     4
## 3 三井     13    10
## 4 宮城      7     7
## 5 流川     11     9
## 6 木暮      5     6

さらに、複数の行を同時に追加する場合には、複数行をもったデータフレームを作成してrbind()で追加する。

Reserve <- data.frame(
  name =c("潮崎","安田","角田","石井","佐々岡","桑田"),
  number=c(8, 6, 9, 12, 13, 15),
  score =c(6, 7, 7,  5,  5,  4)
)
Shohoku <-rbind(Shohoku, Reserve)
Shohoku
##      name number score
## 1    赤木      4     7
## 2    桜木     10     4
## 3    三井     13    10
## 4    宮城      7     7
## 5    流川     11     9
## 6    木暮      5     6
## 7    潮崎      8     6
## 8    安田      6     7
## 9    角田      9     7
## 10   石井     12     5
## 11 佐々岡     13     5
## 12   桑田     15     4

列名が異なると以下の通りにエラーとなる。

Reserve2 <- data.frame(
  name2 =c("潮崎","安田","角田","石井","佐々岡","桑田"),# nameがname2となっている
  number=c(8, 6, 9, 12, 13, 15),
  score =c(6, 7, 7,  5,  5,  4)
)
Shohoku <-rbind(Shohoku, Reserve2)
## Error in match.names(clabs, names(xi)):  名前が以前の名前と一致しません
Shohoku
##      name number score
## 1    赤木      4     7
## 2    桜木     10     4
## 3    三井     13    10
## 4    宮城      7     7
## 5    流川     11     9
## 6    木暮      5     6
## 7    潮崎      8     6
## 8    安田      6     7
## 9    角田      9     7
## 10   石井     12     5
## 11 佐々岡     13     5
## 12   桑田     15     4

2.6.9 行の削除

特定の行を削除したい場合には、以下のように削除したい行番号をベクトルで与えた上で-をつけてやる。

Shohoku <- Shohoku[-c(6,7,8,9,10,11,12), ]
Shohoku
##   name number score
## 1 赤木      4     7
## 2 桜木     10     4
## 3 三井     13    10
## 4 宮城      7     7
## 5 流川     11     9

注意すべき点として、行指定の後に,をつけるのを忘れないようにすること。 もし,をつけ忘れた場合、それは列ベクトルへのアクセスを意味するようになる。今回の場合だと列数は3つしかないので特に何も起こらない。しかし、指定した列番号に対応するだけの列数があった場合には、それらの列が削除されたデータが返ってきてしまい、特にエラーにもならないので間違いに気づきにくくなる。

Shohoku <-rbind(Shohoku, Reserve)#一旦元のデータに戻す
Shohoku <- Shohoku[-c(6,7,8,9,10,11,12) ] # 行番号を指定した後、","をつけ忘れている 
Shohoku
##      name number score
## 1    赤木      4     7
## 2    桜木     10     4
## 3    三井     13    10
## 4    宮城      7     7
## 5    流川     11     9
## 6    潮崎      8     6
## 7    安田      6     7
## 8    角田      9     7
## 9    石井     12     5
## 10 佐々岡     13     5
## 11   桑田     15     4

なお、上記のように連続した番号を指定する場合に、一々すべての番号を記載するのは面倒である。このような場合には、:(シーケンス演算子)を使って始端と終端を記述するとよい。

Shohoku
##      name number score
## 1    赤木      4     7
## 2    桜木     10     4
## 3    三井     13    10
## 4    宮城      7     7
## 5    流川     11     9
## 6    潮崎      8     6
## 7    安田      6     7
## 8    角田      9     7
## 9    石井     12     5
## 10 佐々岡     13     5
## 11   桑田     15     4
Shohoku <- Shohoku[-c(6:12), ]
Shohoku
##   name number score
## 1 赤木      4     7
## 2 桜木     10     4
## 3 三井     13    10
## 4 宮城      7     7
## 5 流川     11     9

2.6.10 データフレームに対する演算

今回のデータフレームには、nameのような文字列型やnumberscore数値型など様々な型のベクトル(列)が含まれている。このような複数の型が混在するデータフレームに対しては演算処理を行うことはできない。演算処理を行いたい場合にはあくまで演算処理をしたい数値型の列のみを抜き出して、演算処理を行う。

Shohoku * 2 #これはエラー
## Error in FUN(left, right):  二項演算子の引数が数値ではありません
Shohoku$score*2 #これは数値型のベクトルなので演算処理が可能
## [1] 14  8 20 14 18
Shohoku$name *2 #これは文字列型のベクトルなのでエラー
## Error in Shohoku$name * 2:  二項演算子の引数が数値ではありません
Shohoku[c("number","score")] *2 #データフレーム型だが、両方とも数値型なので演算処理が可能 
##   number score
## 1      8    14
## 2     20     8
## 3     26    20
## 4     14    14
## 5     22    18

2.6.11 列名へのアクセスと変更

列名を把握したい(アクセスしたい)場合にはcolnames()関数を用いる。 また、列名を変更したい場合には、colnames()でアクセスした後に代入演算子を用いて、新しい列名を与えればよい。

colnames(Shohoku)
## [1] "name"   "number" "score"
colnames(Shohoku) <- c("名前", "背番号", "得点") 
Shohoku
##   名前 背番号 得点
## 1 赤木      4    7
## 2 桜木     10    4
## 3 三井     13   10
## 4 宮城      7    7
## 5 流川     11    9

colnames()は列名をベクトルで返してくるので、[]を用いてインデックス番号を指定してやることによって、個別の列名にアクセスしたり、個別の列名を変更することもできる。

Shohoku
##   名前 背番号 得点
## 1 赤木      4    7
## 2 桜木     10    4
## 3 三井     13   10
## 4 宮城      7    7
## 5 流川     11    9
colnames(Shohoku)[3] <- "score"
Shohoku
##   名前 背番号 score
## 1 赤木      4     7
## 2 桜木     10     4
## 3 三井     13    10
## 4 宮城      7     7
## 5 流川     11     9

自分で作れる程度のデータフレームであればあまり使うことは無いが、後で述べるように、外部のファイルからデータを読み込んだ場合には、そのファイルにどのような列が含まれているのかを把握したり、列名を変更したりするケースはよくある。

2.7 データの読み込み

実際の調査結果のデータを扱うときには、data.frameを毎度自分でちまちま作ってなんていられないので、表計算ソフトで作成されたデータファイルを読み込むことになる。 以下では test.csvtest.xlsxを用いるので、それぞれファイル名をクリックしてダウンロードしておくこと。

2.7.1 データのアップロード

Posit Cloudを利用している場合、Rにデータを読み込ませるには、データをPosit Cloudにアップロードする必要がある。

画面の右下のペインのFileというタブをクリックすると、現在のプロジェクトのフォルダ構成が表示される。

Fileタブ
Fileタブ

このタブの上部の「Upload」をクリックすると、ファイル選択画面が出てきて、データファイルを選択するとファイルがアップロードされる。

Upload
Upload

アップロードしたファイルはFileタブのリストに表示される(今回はtest.csvというファイルをアップロードした)。

Upload
Upload

2.7.2 CSVデータの読み込み

もっともシンプルな方法は表計算ソフトでデータファイルを保存する際にCSVというファイル形式で保存しておき、それをRで読み込む方法である。

CSVファイルとは、行ごとに要素が,(コンマ)で区切られたファイル形式である。例えば、先ほどの表の

名前 背番号 点数
赤木 4 7
桜木 10 4
三井 14 10
宮城 7 7
流川 11 9

の場合、CSVファイルでは以下のようになる。

名前, 背番号, 点数
赤木, 4, 7
桜木, 10,4
三井, 14,10
宮城, 7, 7
流川, 11, 9

このようなファイル形式をRで読み込む場合には、read.csv()関数を使う。1つ目のオプションダブルクォーテーションで括った上で読み込むファイル(既にPosit Cloudにアップロードしてある前提)を指定する。2つ目のオプションでは、最初の行をヘッダー、つまり各列の名前として読み込むのか、最初の行をデータとして読み込むのかの指定である。TRUEだとヘッダーとして読み込み、FALSEだとデータとして読み込む。以下の例では、test.csvというファイルを読み込み、その内容をmyDataというオブジェクトに格納している。

myData <- read.csv("test.csv", header = TRUE)
## Error in make.names(col.names, unique = TRUE):  '<96><bc><91>O' に不正なマルチバイト文字があります
myData
## Error in eval(expr, envir, enclos):  オブジェクト 'myData' がありません

なお、ExcelでCSVファイルを作った場合には、上記のコマンドではうまく読まない場合がある。その場合には3つ目のfileEncodingのオプションを以下の通りに指定する。(上のコマンドでうまく行っている場合には、逆に以下のコマンドだとエラーとなる)

myData <- read.csv("test.csv", header = TRUE, fileEncoding = "Shift-JIS") 
myData
##   名前 背番号 点数
## 1 赤木      4    7
## 2 桜木     10    4
## 3 三井     14   10
## 4 宮城      7    7
## 5 流川     11    9

ちなみに、header=FALSEだと以下の通りとなる。

myData <- read.csv("test.csv", header = FALSE, fileEncoding = "Shift-JIS") 
myData
##     V1     V2   V3
## 1 名前 背番号 点数
## 2 赤木      4    7
## 3 桜木     10    4
## 4 三井     14   10
## 5 宮城      7    7
## 6 流川     11    9

それぞれの列名がV1, V2, V3となっており(自動的にそういう名前が割り当てられる)、もともと列名としていた名前, 背番号, 点数がデータの1行目に含められてしまっているのがわかる。

1列目をヘッダとして読むのか、データとして読むのかは読み込みたいファイルの中身に合わせて設定する。また、1列目をデータとして読み込んだ際には、自分で分かりやすい名前をcolnames()関数を使って設定すること。

2.7.3 Excelデータの読み込み

Excelファイルを直接読み込みたい場合には、Posit Cloudに読み込みたいExcelファイルをアップロードしたうえで、RStudioのFileメニュー->Import Database ->From Excelを選択する。

Excelファイルの読み込み
Excelファイルの読み込み

そのプロジェクト内で初めてExcelファイルを読み込む場合には、追加パッケージの読み込み確認の画面が出てくるが、そのまま「OK」をクリックすればよい。 すると、パッケージの読み込みが暫く行われたのち、以下のようなウィンドウが表示される。

Excelファイルの読み込み画面
Excelファイルの読み込み画面

この画面の右上部の「Browse」ボタンをクリックすると、現在のプロジェクトにあるファイルリストが表示されるので、アップロードしておいたExcelファイルを選択する。すると、自動でファイルの中身が読み込まれ、プレビューが表示される。

このプレビューの下部にあるオプションを指定することによって、読み込みにあたっての細かな設定をする事が出来る。それぞれ、以下の通りである。

項目 内容
Name 読み込んだデータを格納するオブジェクトの名前の設定。デフォルトでは読み込んだファイル名がそのまま入力されている
Sheet Excelファイルのどのシートのデータを読むのかの設定
Range 読み込む範囲の指定。エクセルの範囲表記の方法で記述する
Max Rows 読み込む最大行数の指定。データの行数よりも大きい数を指定した場合には無視される
Skip 読み込みをスキップする行数。指定した行数分だけデータの読み込みが飛ばされる。なお、ヘッダ部はカウントされない
First Row as Names 読み込むにあたって最初の行をヘッダ(列名)として読むのかデータとして読むのかの指定
Open Data Viewer データの読み込みが終わったあと、自動的にデータ閲覧画面を開くかどうかの指定

以下の例では、オブジェクトの名前をmyDataに変更したものである。

オプション設定
オプション設定

右下部には、設定したオプションでExcelファイルを読み込むための関数一式(スクリプトと呼ぶ)が表示されている(このスクリプトは後で重要になるので、ここに表示されているということは覚えておこう)。

この状態で「Import」を押すと、先ほど表示されていたスクリプトが実行され、以下の図のように読み込んだデータの閲覧画面が表示される。

読み込み後
読み込み後

また、以下のようにConcoleにmyDataと入力すると、console中にmyDataの中身が表示される。

myData  # print(myData)でも可
##     V1     V2   V3
## 1 名前 背番号 点数
## 2 赤木      4    7
## 3 桜木     10    4
## 4 三井     14   10
## 5 宮城      7    7
## 6 流川     11    9

2.7.4 データのアクセス、加工

読み込んだデータはデータフレーム型となるため、データフレームの中で説明した方法で、以下のようにして各列にアクセスしたり、データを書き換えたり、列を追加したりできる。

myData <- read.csv("test.csv", header = TRUE, fileEncoding = "Shift-JIS") 
myData$名前
## [1] "赤木" "桜木" "三井" "宮城" "流川"
myData[1, "名前"] <- "Akagi"
myData
##    名前 背番号 点数
## 1 Akagi      4    7
## 2  桜木     10    4
## 3  三井     14   10
## 4  宮城      7    7
## 5  流川     11    9
myData$名前[1] <- "赤木"
myData
##   名前 背番号 点数
## 1 赤木      4    7
## 2 桜木     10    4
## 3 三井     14   10
## 4 宮城      7    7
## 5 流川     11    9
myData$Yomi <- c("あかぎ", "さくらぎ", "みつい", "みやぎ", "るかわ")
myData
##   名前 背番号 点数     Yomi
## 1 赤木      4    7   あかぎ
## 2 桜木     10    4 さくらぎ
## 3 三井     14   10   みつい
## 4 宮城      7    7   みやぎ
## 5 流川     11    9   るかわ

2.8 スクリプト

これまでは命令を1行1行Console中に入力してきた。しかし、色々な処理をさせたり、複雑な処理をさせたりする場合には、命令を1つ1つconsole中に入力して実行していくよりも、命令を別ファイルとして入力し、一気に実行(あるいはある部分だけを実行)する方が作業効率が良い。またそうしておけば、一連の命令をファイルとして保存して、別な時に再度呼び出すことも用意となる。

このような一連の命令セットのことそスクリプトと呼び、それを保存したファイルをスクリプトファイルと呼ぶ。

2.8.1 スクリプトファイルの作成

Rのスクリプトファイルを作成するには、File メニュー-> New File ->R Scriptを選択する。

スクリプトファイルの作成
スクリプトファイルの作成

すると、以下のようなスクリプト編集画面が開く。この画面上で一連のコマンドを入力していく。

実際にスクリプトを入力した例を以下に示す。なお、このスクリプトではExcelのデータの読み込みから始めている。Excelデータの読み込みは、先ほどはImportボタンを押す前のプレビュー画面の右下部にスクリプトのプレビューも表示されていた。あの部分をマウスでコピーし、スクリプト編集画面で貼り付けるとよい。

スクリプトの例
スクリプトの例

2.8.2 スクリプトの実行

作成したスクリプトを実行する際には、実行したい部分をマウスで選択したうえで、編集画面上の「Run」ボタンをクリックするか、キーボードでCtrlキーを押しながらEnterキーを押せばよい。

全体を選択して「Run」もしくはCtrl+Enterすると、スクリプト全体が実行される。

マウスでどの部分も選択せずに「Run」もしくはCtrl+Enterすると、入力カーソルが置かれている行のスクリプトが1行ずつ実行される。

2.8.3 コメントアウト

すでに何度も出てきているが、コマンドを入力するときに#を記載すると、#以降の記述は全て無視される。この#記号はコメントアウト記号といい、スクリプトにコメントやメモを残すのに利用する。スクリプトを作成するときには、是非とも後から別の人がそのスクリプトを読んで内容が理解できるように、メモを残すようにしてほしい。

2.8.4 スクリプトの保存

作成したスクリプトは画面上部の「保存」マークをクリックするか、Ctrl+sキーで保存することができる。Posit Cloud上ではその時のプロジェクト全体の状態が自動的に保存されていくので、つい忘れがちだが、何かのはずみでスクリプトを保存しないまま閉じてしまうと、当然ながらそのスクリプトは帰ってこなくなるので、こまめにファイル保存するのを忘れないようにしてほしい。

保存すると、Posit Cloud上のリストにスクリプトファイルが作成されて、リストに記載される。

保存されたスクリプトファイル
保存されたスクリプトファイル

2.9 ファイルのダウンロード

作成した一連のファイルはあくまでオンライン上に置かれている。手許のパソコンにファイルをダウンロードしたい場合には、左側の「Posit Cloud」と書かれたサイドバーの中の「Your Workspace」をクリックして、プロジェクト管理画面を開く。もしサイドバーを閉じていた場合には、左上にある三本線のマークをクリックするとサイドバーが現れる。

サイドバー
サイドバー

このプロジェクト管理画面の各プロジェクト名の右手にある下向きの矢印のアイコンをクリックすると、プロジェクト全体を圧縮したZipファイルをダウンロードできる。

プロジェクト管理画面
プロジェクト管理画面

ダウンロードしたZipファイルを解凍すると、Posit Cloud上のファイル一式が手許のPCに展開される。

解凍したファイル一式
解凍したファイル一式

2.10 オブジェクトの削除

作ったオブジェクトはRStudio上の右上ペインの「Environment」タブで確認することができる。

オブジェクトリスト
オブジェクトリスト

オブジェクトはいくつでも作ることができるが、あまりに多くなってくるとそれだけPCのメモリを喰ってしまうことになる。 そういった場合には作ったオブジェクトの中で使わなくなったオブジェクトを削除すると良い。 削除する方法は大きく2つある。 1つは、すべてのオブジェクトをまとめて削除する方法で、これは、単にEnvironmentタブの中にあるほうきのアイコンをクリックすればよい。

オブジェクトリスト 一方、特定のオブジェクトを削除したい場合には、Consoleでrm()関数を実行する。関数の引数には削除したいオブジェクトの名前を入力すればよい。

a<- "test1"
b<- "test2"
c<- "test3"
ls() # 今あるオブジェクトを確認するコマンド。別に右上の「Environment」で確認できるが・・・
##  [1] "a"            "b"            "c"            "Full"         "hoge"        
##  [6] "hoge2"        "hoge3"        "kogure"       "my_res"       "my_res.power"
## [11] "myData"       "myName"       "namedhoge"    "NamedScore"   "Res"         
## [16] "Reserve"      "Reserve2"     "Score"        "Shohoku"      "Starting"
rm(a) # aオブジェクトを除く
ls()
##  [1] "b"            "c"            "Full"         "hoge"         "hoge2"       
##  [6] "hoge3"        "kogure"       "my_res"       "my_res.power" "myData"      
## [11] "myName"       "namedhoge"    "NamedScore"   "Res"          "Reserve"     
## [16] "Reserve2"     "Score"        "Shohoku"      "Starting"

aオブジェクトが除かれているのが確認できるだろう。

複数のオブジェクトを削除したい場合には、それらを書き込めばよい。rm()関数については、c()で括る必要はない。

rm(b,c)
ls()
##  [1] "Full"         "hoge"         "hoge2"        "hoge3"        "kogure"      
##  [6] "my_res"       "my_res.power" "myData"       "myName"       "namedhoge"   
## [11] "NamedScore"   "Res"          "Reserve"      "Reserve2"     "Score"       
## [16] "Shohoku"      "Starting"

bcが除かれたのが確認できる。

なお、rm()をスクリプトに書き入れても構わないが、スクリプトに書き入れると、そのスクリプトを実行する度にオブジェクトの削除が行われることになるので、必要に応じてConsoleで実行するのが良いだろう。

3 記述統計量と推測統計量

3.1 記述統計量

あるデータセットの特徴を把握するときに用いられる統計量。元のデータから算出される。なお、ここではデータとは数値のみで構成されているものとする。

以下では実際に以下のデータを使って例を示していく。

Shohoku <- data.frame(
  name = c("赤木","桜木","三井","宮城","流川","木暮","潮崎","安田","角田","石井","佐々岡","桑田"),
  number=c(4,10,13,7,11,5, 8, 6, 9, 12, 13, 15),
  score= c(7,4,10,7,9,6,6, 7, 7,  5,  5,  4)
)
Shohoku
##      name number score
## 1    赤木      4     7
## 2    桜木     10     4
## 3    三井     13    10
## 4    宮城      7     7
## 5    流川     11     9
## 6    木暮      5     6
## 7    潮崎      8     6
## 8    安田      6     7
## 9    角田      9     7
## 10   石井     12     5
## 11 佐々岡     13     5
## 12   桑田     15     4

3.1.1 代表値(平均、中央値、最頻値)

あるデータセット全体的な特徴を一番端的に表現する値。あるいは、そのデータセットを文字通り「代表」する値。平均、中央値、最頻値の3種類がある。

平均値

すべてのデータ(数値)を足し合わせて、データの数で割った値のこと。

\[ \bar{x} =\frac{1}{n}\sum^{n}_{i=1}x_{i} \] Rで求めるにはmean()関数を用いる。(各自、スクリプトファイルを作成し、以下の内容を自分でも入力・実行して出力結果を確認すること)

example<-c(1,2,3,4,5) # 例として1,2,3,4,5という5つの数値からなるベクトルを使った
mean(example)
## [1] 3
mean(Shohoku$score) #先ほどの湘北高校バスケ部のフリースロー10本勝負のスコアの平均
## [1] 6.416667

なお、実際のデータを扱う際には、CSVファイルやExcelファイルからデータを読み込むことが多くなる。そうしたデータファイルにはたまに「空白」が含まれているケースがある(例えばアンケートで無回答だったり、実験で計測に失敗したりした場合)。その場合、Rにそれらのデータを読み込むと、空白だった箇所にはNA(Not Available:利用不可能という意味)という記号が自動的に入力される。

mean()関数を用いる際に、与えたベクトルにNAが含まれていた場合、Errorにはならないが出力はNAとなる。

このようなデータの場合には、以下のようにna.omitオプションをTrue(真)に設定するとNAを除いた上で平均が算出される(データの総数はNAの数の分だけへることになる。)。

example<- c(1,2,3,4,NA)
mean(example)
## [1] NA
example<- c(1,2,3,4,NA)
mean(example, na.rm = T) 
## [1] 2.5

中央値

全てのデータ(数値)を数の大小で順に並べ、ちょうど中央の順位に来る数値。データ点数が偶数だった場合には、中央の位置の前後の数字の平均値を中央値とする。

Rではmedian()を用いる

median(Shohoku$score)
## [1] 6.5

先の湘北高校のデータは12人分なので、中央値は6位と7位の間をとることになるため、小数の値が出てきている。

最頻値

全てのデータ(数値)について、度数分布をとり、最も度数が大きい(出現頻度が最も多い)数値。

Rでは最頻値を直接求める関数は存在しない。そのため、まずはtable()コマンドを使って、対象となるデータの度数分布表を求める。 ついで、その度数分布表に度数が最も高いものを求める、という手続きで算出する。 ただし、度数分布表で最も度数が高いものを取り出したときのデータは「名前付きベクトル」{#名前付きベクトル}で出力され、最頻値を表す値は度数値に対する「名前」(すなわち文字列)として出力れる。そこでその名前を数値として読みなおす必要がある。 以下は例である。行っている一連の処理をよく確認しておくこと。

table_data<-table(Shohoku$score) ## 度数分布表を作成
print(table_data)
## 
##  4  5  6  7  9 10 
##  2  2  2  4  1  1
d<-which.max(table_data) ##度数分布表から最も度数が高い値を取り出す
print(d) # 7と4が表示されている。わかりにくいが、これは"7"という名前の付いた列に4という度数データが入っている、ということを表している。ここで"7"は数値ではなく文字列である。
## 7 
## 4
mode<-as.numeric(names(d)) #欲しいのは値としての7なので、names()を掛けた後に、as.numeric()を行っている。
print(mode)
## [1] 7

ちなみに、上記のスクリプトファイルは細かく一つ一つ分けて書いているが、以下のように関数の中に関数を入れることで、シンプルに書くことができる。このように関数が入れ子になっている場合、内側の関数から順次実行されている。

as.numeric(names(which.max(table(Shohoku$score))))
## [1] 7

さらには、tidyversというライブラリを読み込んでおくことによって、以下のような書き方もできる。なお、初めてTidyverseを使う場合には、library(tidyverse)を読み込んだときにパッケージのインストールをするかどうか聞かれるので、installを選択しておく。tidyverseについては後ほど別章で詳しく説明する。

library(tidyverse)
Shohoku$score %>%
  table() %>%
  which.max() %>%
  names() %>%
  as.numeric() 
## [1] 7

平均・中央値・最頻値の関係

正規分布のように左右対象の一山の度数分布をしていた場合には、平均値・中央値・最頻値は全てほぼ一致する。

左右対称の一山の分布(正規分布)
左右対称の一山の分布(正規分布)

一方、次の図のように左右非対称であった場合には、平均値・中央値・最頻値は一致しなくなる。

左右非対称の分布
左右非対称の分布

3.1.2 ばらつき(分散・標準偏差、四分位数)

あるデータセットがどの程度ばらつき(広がり)を持っているのかを示す指標。平均系の平均偏差・分散・標準偏差と、中央値系の四分位数とがある。

平均偏差

平均偏差とは、データ全体の平均値を予め求めておいた上で、各データと平均値との距離(差の絶対値)を全てのデータに対して求め、その平均をとったもの。

平均からの距離
平均からの距離

\[ M.A.D =\frac{1}{n}\sum^{n}_{i=1}|x_{i} - \bar{x}| \] Rではmad()関数を用いる。

MAD <- mad(Shohoku$score)
print(MAD)
## [1] 1.4826

分散

平均偏差は数式のなかに絶対値記号が入ってきて、数値計算が非常に面倒になる。絶対値を加えたのは単純に\(x_{i}-\bar{x}\)を求めると\(x_{i}\)によっては負の数字になるからである。負を正にするためだけなら、\(x_{i}-\bar{x}\)を2乗するということでもよいのではないか。そこで、「差の絶対値」ではなく「差の2乗」の平均値をばらつきの指標としたものが分散(Variance)である。一般に \(S^{2}\)と表現する。「2乗の平均」にしたことによって数値自体の絶対的な意味の解釈は難しくなるが、少なくとも2つのデータセットがある時に、どちらのデータのほうがよりばらついているかは分散の値を比較することによって分かる。

\[ S^{2} = \frac{1}{n}\sum^{n}_{i=1}(x_{i}-\bar{x})^2 \]

Rではvar()関数とnrow()関数を用いて以下のように書く。nrow()はデータフレームの行数、すなわちデータフレームに含まれているサンプル数を返す関数である。

s2<- var(Shohoku$score)
n <- nrow(Shohoku)
VAR <- (n-1)/n*s2
print(VAR) #こちらが標本分散
## [1] 3.076389

複雑な式に見えるが、要するに、\(\frac{データ数-1}{データ数}\)var()に掛けているということである。これはvar()関数は不偏分散を算出する関数であり、データそのものの分散(標本分散)を算出するためには、補正をしてやる必要があるためである。不偏分散については後ほど改めて説明する。

標準偏差

分散の場合、距離(差)の2乗にしてしまっているので、元のデータと次元が異なってしまい、出てきた数値が何を意味しているかがにわかには分からない。そこで分散の平方根を取ることによって、元のデータと同じ次元の数値とすることによって、元のデータと比較したり平均と組み合わせられるようにしたものが標準偏差(Standard Deviation)である。一般に\(S\)と表現する。

\[ S=\sqrt{S^{2}} = \sqrt{\frac{1}{n}\sum^{n}_{i=1}(x_{i}-\bar{x})^2} \]

Rではsd()関数を用いるが、データそのものの標準偏差を得たい場合には分散の時と同様にnrow()関数を用いた補正(標準偏差の場合にはさらに平方根を求める関数sqrt()も用いる)をしてやる必要がある。

s <- sd(Shohoku$score) # こちらのsは小文字
S <- sqrt((n-1)/n)*s # nは先に求めたnrow()関数の出力。
print(S)
## [1] 1.753964

標準偏差は「ばらつき」の単位(ものさし)として用いることが一般的である。データが完全に正規分布をしていた場合、次の図のように、平均値から±1Sだけ離れたところに境界を引くと、その境界内には全データの約68.27%のデータが含まれる。同様に±2Sの範囲内であれば約95.45%のデータが、±3Sになると約99.73%のデータが含まれることになる。

標準偏差とデータの分布
標準偏差とデータの分布

データの標準化(正規化)得点

平均\(\bar{x}\)を基準(すなわち0)とし、それぞれのデータの値\(x_{i}\)と平均との距離を標準偏差\(S\)を単位として表現した値。要するに、そのデータ全体の中で、あるデータ値が相対的にどのくらいの位置にあるのかを表した値のことを標準化得点、もしくは正規化得点と呼ぶ。一般に\(z_{i}\)と表現する。さらに、こうした処理を行うことを標準化、あるいは正規化と呼ぶ。 \[ z_{i} = \frac{x_{i}-\bar{x}}{S} \]

Rで標準化を行う場合、scale()関数を用いればよい。

z <- scale(Shohoku$score)
print(z)
##             [,1]
##  [1,]  0.3184211
##  [2,] -1.3191733
##  [3,]  1.9560156
##  [4,]  0.3184211
##  [5,]  1.4101508
##  [6,] -0.2274437
##  [7,] -0.2274437
##  [8,]  0.3184211
##  [9,]  0.3184211
## [10,] -0.7733085
## [11,] -0.7733085
## [12,] -1.3191733
## attr(,"scaled:center")
## [1] 6.416667
## attr(,"scaled:scale")
## [1] 1.831955

出力結果の上12個は各データ値を標準化した値である。続いて、attr()にあるのは、それぞれ標準化に用いた平均値と標準偏差である。先ほど求めた平均、標準偏差と比べてみると、平均は一致しているが標準偏差は異なる。これは、scale()関数が内部で計算に用いている標準偏差は、あくまで補正の掛けない不偏標準偏差、すなわちsd()関数の結果をそのまま計算に用いているからである。実際に、先ほどsd()関数の結果を格納したs(小文字)をプリントさせてみると、同じ値が出力される。

print(s)
## [1] 1.831955

偏差値

標準化得点が得られれば、受験の際にお世話になった「偏差値」というものを以下の式で算出することができる。 \[ 偏差値= z_{i}*10+50 \] すなわち、偏差値とは標準化得点を10倍したものに50を足したものである。この式から得点が平均点と一致していた場合に偏差値が50になることが理解できる。また、データが正規分布をしている場合には、先に示した標準偏差とデータの分布の図より、偏差値が60から40の範囲内に約68%のデータが含まれていることがわかる。さらに、偏差値が70を超えていた場合には、そのデータ値は全体の上位5%以内にある、ということになる。

四分位数

全てのデータ(数値)を小さいものから順に並べ、最小値(0%)、下位25%、中央値(50%)、上位25%、最大値(100%)の順位に来る数値。特に下位25%を第1四分位数、上位25%を第3四分位数と呼ぶ。 分散と標準偏差が平均値を代表値とした場合のばらつきの指標であるのに対して、中央値を代表値とした場合のばらつきの指標が四分位数となる。

四分位数を求めるRの基本関数はquantile()である。quantile()関数は算出対象となるデータを与えた後に、どの四分位数を求めるかを小数で与えてやる。

Q_1 <- quantile(Shohoku$score, 0.25) # 第1四分位数 下位25%の値。
Q_2 <- quantile(Shohoku$score, 0.75) # 第3四分位数 上位25%、すなわち下から75%の値
print(Q_1)
print(Q_2)
## 25% 
##   5 
## 75% 
##   7

このように関数を用いるときに()に与えるデータやオプションのことを引数(ひきすう)と呼ぶ。

また、quantile()は以下のように2つ目の引数にc()を用いて複数のパラメータを指定してやることで、複数の四分位数を一度に求めることもできる。

Q <- quantile(Shohoku$score, c(0.0, 0.25, 0.50, 0.75, 1.0))
print(Q)
##   0%  25%  50%  75% 100% 
##  4.0  5.0  6.5  7.0 10.0

ちなみに、quantile()の出力は名前付きベクトルである。

四分位数はその数字そのものを示すより、一般に「箱ひげ図」と呼ばれるグラフを作成する際に用いられることが多い。箱ひげ図の具体例を以下に示す。箱の上端と下端で第1、第3四分位数を示し、箱の中の線で中央値を示す。さらに上下に伸びたひげで最大値、最小値を示す。場合によっては、箱の中に点線で平均値を示したり、一点鎖線で最頻値を示したりといったことも可能である。

箱ひげ図
箱ひげ図

このグラフの秀逸な点は、データが概ねどのような分布をしているかを示すことができることである。例えば上の箱ひげ図の場合、上にひげが長く伸び、さらに第3四分位数と中央値の間が、第1四分位数と中央値の間よりも広がっていることから、データの分布として、全体として点の低いほうに山が偏っており、点の高いほうに裾野が伸びた分布をしているということが読み取れる。 以前は棒グラフで平均値を示したグラフや、平均値の棒グラフに標準偏差の幅分だけ「ひげ」(エラーバーと呼ぶ)をつけたグラフを示すのがほとんどであったが、平均と標準偏差を示すだけでは、上記のような分布の形状までは分からない。このため、最近では、箱ひげ図を示すことも多くなってきている。

3.1.3 記述統計量をまとめて出力するコマンド

以上、個別に記述統計量を算出する方法について説明した。 算出したこれらの統計量を別の計算で用いたい場合には、個別にこれらのコマンドで算出する必要がある。 一方、レポートや論文などで記述統計量を報告したり、とりあえず全体的な傾向を把握したいといった場合には、summary()関数やdescribe()関数を利用することができる

summary()関数

summary()関数を使うと、ベクトルやデータフレームに含まれている各列について、四分位数と平均値を一度に得ることができる。また、ベクトルやデータフレームの列が文字列型だった場合にはデータの点数が、ベクトルやデータフレームの列が要因型(factor型)だった場合には要因ごとのデータ点数が表示される。

簡単にデータ全体の傾向を把握することができるので、データを読み込んだ場合にはsummary()をまずは実施する、というのがデータ分析の基本フローと言えるだろう。

summary(Shohoku) #データフレームを与えると、含まれているベクトルごとの平均と四分位数を出力される。
##      name               number           score       
##  Length:12          Min.   : 4.000   Min.   : 4.000  
##  Class :character   1st Qu.: 6.750   1st Qu.: 5.000  
##  Mode  :character   Median : 9.500   Median : 6.500  
##                     Mean   : 9.417   Mean   : 6.417  
##                     3rd Qu.:12.250   3rd Qu.: 7.000  
##                     Max.   :15.000   Max.   :10.000
summary(Shohoku$score) #ベクトルを与えると、そのベクトルの平均と四分位数をが出力される。
##    Min. 1st Qu.  Median    Mean 3rd Qu.    Max. 
##   4.000   5.000   6.500   6.417   7.000  10.000

describe()関数

psychパッケージに含まれるdescribe()関数を用いるとより詳しい記述統計量を得ることができる。psychパッケージを利用するためには、上部のメニューの中からTools->install.packagesを選択したうえで、以下のようにPackagesにpsychと入力してパッケージをインストールする。

パッケージのインストール
パッケージのインストール

インストールが完了すれば、library()関数を使ってpsychパッケージを読み込み、describe()関数を実行すると、以下のような結果が表示される。

library(psych)
## 
##  次のパッケージを付け加えます: 'psych'
##  以下のオブジェクトは 'package:ggplot2' からマスクされています:
## 
##     %+%, alpha
describe(Shohoku)
##        vars  n mean   sd median trimmed  mad min max range  skew kurtosis   se
## name*     1 12 6.50 3.61    6.5     6.5 4.45   1  12    11  0.00    -1.50 1.04
## number    2 12 9.42 3.50    9.5     9.4 4.45   4  15    11 -0.03    -1.47 1.01
## score     3 12 6.42 1.83    6.5     6.3 1.48   4  10     6  0.41    -0.88 0.53

それぞれ、以下のような指標である

名前 内容
mean 平均
sd 標準偏差(不偏標準偏差)
median 中央値
trimmed トリム平均(データの上下5%ずつを取り除いて算出した平均)
mad 平均偏差
min 最小値
max 最大値
range 最大値-最小値
skew 歪度(データの分布がどの程度左右に偏っているか。負の場合は左に、正の場合には右に偏っている。また値が大きいほど偏りが大きい。)
kurtosis 尖度(データの分布がどの程度尖っているか。正規分布の尖り具合を0として、正の場合には正規分布より尖っており、負の場合には正規分布より平坦になっている)
se 標準誤差(詳しくは後述{#平均値の区間推定}。)

なお、上記のname列に対する結果の通り、describe()関数はたとえ列の型が文字列であっても数字を出力してくる。当然ながら意味は全くない。参考までに、この出力結果は各要素のインデックス番号をデータとして計算を行ったものであある。

この例のようにRでは標準的なコマンド以外にも様々なコマンドが用意されている。ただし、それらのコマンドを利用するためには、そのコマンドが定義された外部パッケージをTools->install.packagesによってインストールし、さらにスクリプト内でlibrary()関数を使ってそのパッケージを読み込む必要がある。 なおlibrary()関数での外部パッケージの読み込みはスクリプトの実行中1回だけで良い。1度読み込みば、その後はその外部パッケージのコマンドを入力するだけでコマンドを実行することができる。 またパッケージのインストールも一度行えばその後は行う必要はなくなる。

3.2 推測統計量

3.2.1 全数調査とサンプリング調査

何らかの調査をするときに、その調査が想定している対象全体のことを「母集団」という。それに対して、母集団の中から一部だけを抜け出して調査を行った場合、実際に調査を行った対象を「サンプル(標本)集団」と呼び、サンプル集団に含まれる個々のデータのことをサンプル(標本)と呼ぶ。また、母集団に対して行う調査を母集団調査、あるいは全数調査と呼び、サンプル集団に対して行う調査をサンプル調査と呼ぶ。

母集団とサンプル
母集団とサンプル

母集団調査(全数調査)を実施できればそれに越したことはないが、実際には費用的・時間的な制約であったり、現実に不可能であったりするために、母集団調査を実施できないことは多い。そういう場合には、現実的に実施可能な限られた数のサンプル調査を行い、その結果をもとに母集団の特徴を推測するということが行われる。

母集団調査(全数調査)の例

  • 中学での校内の定期テスト:あくまでその中学の中での成績を把握するためのものであり、対象はその中学の生徒に限られ、さらに全員が受けることが前提のテストであることから、母集団にアクセスできていると言える。同様に大学入試共通テストもその年度の全受験生が受けている(もちろん受けない受験生もいないわけではないが、そのような受験生はそもそも対象とはなっていない)ので、これも母集団全体にアクセスできている例である。
  • 国勢調査:国勢調査はその時の日本の状態を把握することを目的に、5年に一回、日本国に在住のすべての人を対象に行われる調査であり、母集団そのもの(その時に日本に在住している人全て)にアクセスしている例である。
  • 小中学校で行われる健康診断:日本の場合、小学校と中学校は義務教育として対象者全員が通うことが義務付けられている。さらに健康診断は学校保健安全法により全ての学校で実施することが定められている。したがって、小中学校における健康診断はその年度の日本の6歳児から15歳児までの全員に対する診断であり、母集団にアクセスしている例である。一方、高校や大学でも全員に対する健康診断は行われているが、これらの学校は義務教育ではない。したがって、これらの学校での健康診断結果は、日本の16歳〜22歳の母集団全体に対する調査結果ではない。

母集団にアクセスできず、あくまでサンプル調査となる例

  • 血液型による性格の違いの検証のための性格診断調査:実際に血液型によって性格が違っているかを全数調査によって検証するためには、全世界の全ての人に調査しなければならない。さらに現時点での全世界の人にアクセスできたとしても「単に今現在はそうなってるだけだろう」というツッコミが入り得るので、理論としての完全性を確保するためには、過去の全人類やこれから先の全人類を調査しなければならない。当然ながらそんなことは不可能である。したがって、血液型による性格の違いを母集団調査によって検証することは不可能であり、どれだけ多くのデータを集めたとしても、それはあくまでサンプル調査である。
  • 新聞社等が実施する世論調査:世論調査は各自治体が持つ選挙人名簿から10人おきや20人おきにピックアップされた人に対してハガキや電話、あるいは訪問によって調査している。得たいのはその時点での日本全体の世論であり、その時の日本人への全数調査ができれば良いが、それをするにはコスト的にも時間的にも現実的ではないので、限られた数のサンプルだけに調査を行い、そこから得た結果をもとに、日本全体の世論を推測している。
  • 視聴率調査:視聴率はビデオリサーチ社がモニター世帯に選ばれた世帯に対して「どの番組を視聴しているか」を記録する機械を設置し、その機械に記録された情報をもとに算出している。あくまでモニター世帯だけしか調査は行なっていないのでサンプル調査であり、その結果をもとに「日本全体でどの番組がどの程度視聴されているか」を推測している。

そのほか、マーケティング調査なども基本的に全てサンプル調査である。

3.2.2 母数と標本統計量

母集団全体での記述統計量を母数と呼ぶ。また、特に平均、分散については母平均母分散と呼ぶこともある。 それに対してサンプル集団の記述統計量を標本統計量と呼び、平均を標本平均、分散を標本分散と呼ぶ。

母集団全体にアクセスできる場合には、得られたデータの記述統計量は母数そのものとなる。それに対してサンプル集団にしかアクセスできない場合には、そのデータの記述統計量はあくまで標本統計量であり母数ではない。

3.2.3 標本統計量からの母数の推測

先の例にも述べた通り、サンプル調査ではデータそのものの記述統計量(つまり標本統計量)が得たいのではなく、その結果を基に母数を推定することが目的である。従って、標本統計量から母数を推定する必要がある。

推定には大きく2種類あり、値を単刀直入に推定することを点推定と呼ぶ。一方、「95%の確率で、この範囲内に母数がある可能性が高い」という範囲を推定することを区間推定と呼ぶ。点推定の値のことを不偏推定量、区間推定の範囲のことを「95%信頼区間」と呼ぶ。なお、区間推定では「95%」が用いられることが多いが、場合によっては90%や99%が用いられることもああり、その場合には信頼区間の呼び名もそれぞれの確率値に変わる。

細かな導出過程や導出に当たっての考え方については省略するが、母数の推定値は以下の数式によって算出される。なお、以下では\(\mu\), \(\sigma^2\)は母平均と簿分散、\(\hat\mu\)\(\hat{\sigma^2}\)は母平均と簿分散の推定量、\(\bar{x}\)\(s^2\)は標本平均、標本分散を表す。

平均値の点推定

母平均の不偏推定量を式で表すと以下のようになる。 \[ \mu \approx \hat{\mu}=\bar{x}=\frac{1}{n}\sum^{n}_{i=1}x_{i} \] 要するに、母平均の推定量は標本平均そのものである。つまり、サンプル調査を行った時、得られたデータの平均を母集団全体の平均と見なそう、ということである。

したがってRで母平均の推定量を求めるときも、従来通りmean()関数を用いればよい。

mean(Shohoku$score)
## [1] 6.416667

分散の点推定

母分散の不偏推定量(不偏分散)は以下の式となる。

\[ \sigma^2 \approx \hat{\sigma^2}=\frac{n}{n-1}s^2=\frac{1}{n-1}\sum^{n}_{i=1}(x_i-\bar{x})^2 \]

要するに、標本分散を求める時には「データと平均との差の二乗の和」をnで割っていたのを、n-1で割る形にしているだけである(なぜn-1になるかは結構面倒な話になるので省略。興味ある人は上の数式の導出過程をネットで調べてみてほしい)。

さらに母標準偏差の不偏推定量は、上記の式の平方根となる。

\[\begin{align} \sigma \approx \hat{\sigma}&=\sqrt{\frac{n}{n-1}s^2} = \sqrt{\frac{1}{n-1}\sum^{n}_{i=1}(x_i-\bar{x})^2} \\ &= \sqrt{\frac{n}{n-1}}s \end{align}\]

Rで不偏分散や標準偏差の不偏推定量を求める方法については、すでに上記で説明したとおり、var()sd()を用いればよい。

すなわち、先ほどの湘北高校のデータを用いるならば、以下の通りとなる。

var(Shohoku$score)
sd(Shohoku$score)
## [1] 3.356061
## [1] 1.831955

ちなみに、この数値の解釈として、湘北高校のバスケ部員はフリースローで10本中、約6本(平均の不偏推定量)を入れることができるが、メンバー間でのばらつきとして1.8点ほどの標準偏差を持っている、ということである。これはあくまで母数の推定量をであり、赤木・桜木・三井・宮城・流川の5人の得点の平均や5人の中でのばらつきではない。5人をサンプルとした「湘北高校のレギュラーメンバー」という母集団のフリースローの成績の推定値ということである。つまり、赤木・桜木・三井・宮城・流川が入学する前のレギュラーメンバーや、彼らが引退した後のレギュラーメンバーも対象に入れた、湘北高校バスケ部のレギュラーメンバー一般の成績の推定値ということになる。

平均値の区間推定

続いて、区間推定であるが、平均の区間推定を行う際には標準誤差(Standard Error)と呼ばれる統計量を算出する必要がある。標準誤差は以下の数式で算出される。

\[ SE = \sqrt{\frac{\hat{\sigma^2}}{n}}=\frac{\hat{\sigma}}{\sqrt{n}} \] 要するに、標準偏差の不偏推定量を\(\sqrt{n}\)で割った値である。 この標準誤差(以下SE)を用いて、平均値の95%信頼区間(上限と下限)はそれぞれ以下の数式で得られる。

\[\begin{align} 上限 &= \hat\mu + t\times SE \\ 下限 &= \hat\mu - t\times SE \end{align}\]

ここで\(t\)とは自由度\(n-1\)\(n\)はサンプル数)のT分布表の両側95%から得られる値である。T分布表は大抵の統計学の教科書の末尾に記載されている。具体的には以下のような表である。自由度n-1の行の\(\alpha=0.025\)の値を読む。例えば、サンプル数が30の場合、自由度は29なので、t=2.045となる。

T分布表
T分布表

Rで平均値の信頼区間を求めるにあたっては、2つの方法がある。

まず一つ目はt.test()を使う方法である。本来t.test()関数は後で説明する平均の比較に用いる関数であるが、信頼区間を出力させるのに用いることができる。

res<-t.test(Shohoku$score)
res$conf.int
## [1] 5.252698 7.580636
## attr(,"conf.level")
## [1] 0.95

\(5.252698\)が下限、\(7.580636\)が上限となる。\(0.95\)は信頼区間の確率である。

ちなみに、res自体を出力させると以下のような出力になる。ごちゃごちゃと書かれているが、真ん中あたりに95 percent confidence interval:とあり、信頼区間が記載されている。

print(res)
## 
##  One Sample t-test
## 
## data:  Shohoku$score
## t = 12.133, df = 11, p-value = 1.038e-07
## alternative hypothesis: true mean is not equal to 0
## 95 percent confidence interval:
##  5.252698 7.580636
## sample estimates:
## mean of x 
##  6.416667

もし信頼区間の確率設定を変えたい場合には以下のようにすればよい。

res<-t.test(Shohoku$score, conf.level = 0.99)
res$conf.int
## [1] 4.774192 8.059141
## attr(,"conf.level")
## [1] 0.99

二つ目の方法は、先に示した式を自分で打ち込む方法である。

# 平均と標準誤差の算出
n <-length(Shohoku$score)#length()はベクトルの長さ(要素数)を得る関数。もちろんnrow(Shohoku)としてもよい。
res_mean <- mean(Shohoku$score)
res_se <- sd(Shohoku$score)/sqrt(n)  

#t値の取得
t_value <- qt(0.975, df=n-1) # 0.025でもよい。0.025とした場合、負の数字が出力される点に注意すること。

#信頼区間の算出
Band_Upper <- res_mean + t_value * res_se
Band_Lower <- res_mean - t_value * res_se

#出力
c(Band_Lower, Band_Upper)
## [1] 5.252698 7.580636

先ほどと同じ数値が算出されていることが確認できる。

分散の区間推定

分散についての95%信頼区間については不偏分散\(\hat{\sigma^2}\)を用いて次の式で与えられる。

\[\begin{align} 上限 &= \frac{(n-1)\hat{\sigma^2}}{\chi^2_{0.975}} \\ 下限 &= \frac{(n-1)\hat{\sigma^2}}{\chi^2_{0.025}} \end{align}\]

\(\chi^2\)(カイ2乗と呼ぶ)は、\(t\)と同様に自由度n-1(nはサンプル数)のカイ2乗分布表から得られる値である。分布表と同様に、カイ2乗分布表も統計学の教科書には記載されている。具体的には次々項のような表である。自由度n-1の\(\alpha=0.025と0.975\)の値を読み取る。例えば、サンプル数が30の場合、自由度は29なので、上限算出に使う\(\chi^2\)値は16.047、下限算出に使う\(\chi^2\)値は45.722となる

\chi^2分布表
\(\chi^2\)分布表

Rで分散の信頼区間を算出する方法も2つある。1つはEnvStatsパッケージに含まれているvarTest()関数を使う方法である。使うにはEnvStatsパッケージをインストールしたうえで、library()関数を実行する必要がある(こちら参照)。

library(EnvStats)
## 
##  次のパッケージを付け加えます: 'EnvStats'
##  以下のオブジェクトは 'package:stats' からマスクされています:
## 
##     predict, predict.lm
res<-varTest(Shohoku$score)
res$conf.int
##      LCL      UCL 
## 1.684151 9.674817 
## attr(,"conf.level")
## [1] 0.95
res<-varTest(Shohoku$score, conf.level = 0.90) #90%信頼区間の場合
res$conf.int
##      LCL      UCL 
## 1.876310 8.069546 
## attr(,"conf.level")
## [1] 0.9

ちなみにvarTest()自体の出力結果は以下の通りである。一番下に信頼区間が記載されている。なお、上のコードで確率を90%に設定したものをresに格納しているので、90%信頼区間の結果が表示されている。

print(res)
## 
## Results of Hypothesis Test
## --------------------------
## 
## Null Hypothesis:                 variance = 1
## 
## Alternative Hypothesis:          True variance is not equal to 1
## 
## Test Name:                       Chi-Squared Test on Variance
## 
## Estimated Parameter(s):          variance = 3.356061
## 
## Data:                            Shohoku$score
## 
## Test Statistic:                  Chi-Squared = 36.91667
## 
## Test Statistic Parameter:        df = 11
## 
## P-value:                         0.0002379912
## 
## 90% Confidence Interval:         LCL = 1.876310
##                                  UCL = 8.069546

もう一つの方法は、平均値の区間推定と同様に先に示した式を自分で打ち込む方法である。

n <- nrow(Shohoku) # 今度はnrow()を用いた。当然length()を用いてもよい
res_var <- var(Shohoku$score)

# カイ二乗分布のクリティカル値を使用して信頼区間を計算
chi_upper <- qchisq(0.025, df = n - 1) #上で示した式の上限・下限の確率値が入れ替わっていることに注意
chi_lower <- qchisq(0.975, df = n - 1) 

# 分散の信頼区間
Band_Lower <- (n - 1) * res_var / chi_lower 
Band_Upper <- (n - 1) * res_var / chi_upper

#出力
c(Band_Lower, Band_Upper)
## [1] 1.684151 9.674817

なお、Rで計算するときの\(\chi^2\)値算出の関数qchisq()での確率設定値と、先ほど式で示したものの中での\(\chi^2\)値の確率設定値が入れ替わっている。これは確率を「残された方」で見るのか、「切り落とす方」で見るのかの違いである。入れ替わってしまったとしても数値自体が変わるわけではないので、実際に出力させてみて、上限と下限の値が逆順になっていれば、数値を入れ替えればよい。

平均値の区間推定にしろ、分散の区間推定にしろ、「95%」信頼区間なのに、\(\alpha=0.05\)\(0.95\)ではなく\(\alpha=0.025\)\(0.975\)の値を読むのはなぜなのか疑問に思う人もいると思う。これは「95%の確率でその区間の中に平均値がある」という範囲を取り出す、ということはその範囲よりも上下にある2.5%ずつを切り落とすということだからである。仮に5%と95%の値を読むことにしてしまうと、残された区間は90%となってしまう。

3.3 グループごとの統計量の算出

これまでつかってきたデータ(湘北高校のデータ)には特にグループ(群)分けという概念は存在していないデータであった。 しかし実際のデータには大抵、属性項目(例えば、性別や、実験データにおいては実験条件、あるいは学年や年齢区分、国籍など)によるグループ分けというものが存在する。そういったグループ分けが存在するときに、グループごとに記述統計量(推測統計量)を算出するには、aggregate()関数を用いると良い。

以下では先の湘北高校のデータを、レギュラー組と控え組にグループ分けしたものにしたうえで、グループごとの平均と標準偏差を算出させてみる。

3.3.1 要因型変数の作成

まずは湘北高校のデータにグルプ分けのための変数Groupを追加する。

Shohoku["Group"] <- as.factor(c("レギュラー", "レギュラー", "レギュラー", "レギュラー", "レギュラー", "控え", "控え", "控え", "控え", "控え", "控え", "控え"))
Shohoku # 確認
##      name number score      Group
## 1    赤木      4     7 レギュラー
## 2    桜木     10     4 レギュラー
## 3    三井     13    10 レギュラー
## 4    宮城      7     7 レギュラー
## 5    流川     11     9 レギュラー
## 6    木暮      5     6       控え
## 7    潮崎      8     6       控え
## 8    安田      6     7       控え
## 9    角田      9     7       控え
## 10   石井     12     5       控え
## 11 佐々岡     13     5       控え
## 12   桑田     15     4       控え

as.factor()関数に関しては先に軽く触れたが(参考)、ここで改めて詳しく説明する。 グループ分けに使う変数の型は「要因型」と呼ぶ。要因型に設定するためにはc()関数を使って作成したベクトルをas.factor()関数で括ってやる必要がある。内容はレギュラー控えという文字列になっているが、列全体としては文字列型ではないことに注意してもらいたい。

文字列型と要因型とでは結果が異なる。summary()の実行結果として出力されるものが異なってくる。それぞれ以下で確認してみてほしい。

Shohoku$Group <- as.factor(c("レギュラー", "レギュラー", "レギュラー", "レギュラー", "レギュラー", "控え", "控え", "控え", "控え", "控え", "控え", "控え"))
summary(Shohoku)
##      name               number           score               Group  
##  Length:12          Min.   : 4.000   Min.   : 4.000   レギュラー:5  
##  Class :character   1st Qu.: 6.750   1st Qu.: 5.000   控え      :7  
##  Mode  :character   Median : 9.500   Median : 6.500                 
##                     Mean   : 9.417   Mean   : 6.417                 
##                     3rd Qu.:12.250   3rd Qu.: 7.000                 
##                     Max.   :15.000   Max.   :10.000
Shohoku$Group <- c("レギュラー", "レギュラー", "レギュラー", "レギュラー", "レギュラー", "控え", "控え", "控え", "控え", "控え", "控え", "控え")
summary(Shohoku)
##      name               number           score           Group          
##  Length:12          Min.   : 4.000   Min.   : 4.000   Length:12         
##  Class :character   1st Qu.: 6.750   1st Qu.: 5.000   Class :character  
##  Mode  :character   Median : 9.500   Median : 6.500   Mode  :character  
##                     Mean   : 9.417   Mean   : 6.417                     
##                     3rd Qu.:12.250   3rd Qu.: 7.000                     
##                     Max.   :15.000   Max.   :10.000

要因型にした場合には、それぞれのグループに含まれるデータの数が返されるのに対して、文字列型の場合には全体のデータ点数と文字列型であることを示すClass: characterMode: charaterという表示が返されるだけである。

なお、要因型に設定するのであれば、その内容は数値でも構わない。ただ、数値をグループ分け変数に使用する場合には特にas.factor()を実行するのを忘れないようにしなければならない。文字列型だった場合には数値演算はできないので結果自体は要因型であってもなくても変わらないことはよくあるが、数値型のときにas.factor()を忘れてしまうと、グループ分け用の変数としていたつもりでも数値の変数として処理され、演算の中に組み込まれてしまうことがあり、誤った結果を出力するケースがある(例えば、回帰分析)。

Shohoku$Group <- as.factor(c(1,1,1,1,1,2,2,2,2,2,2,2)) #1:レギュラー, 2:控え
summary(Shohoku)
##      name               number           score        Group
##  Length:12          Min.   : 4.000   Min.   : 4.000   1:5  
##  Class :character   1st Qu.: 6.750   1st Qu.: 5.000   2:7  
##  Mode  :character   Median : 9.500   Median : 6.500        
##                     Mean   : 9.417   Mean   : 6.417        
##                     3rd Qu.:12.250   3rd Qu.: 7.000        
##                     Max.   :15.000   Max.   :10.000
Shohoku$Group <- c(1,1,1,1,1,2,2,2,2,2,2,2)
summary(Shohoku)
##      name               number           score            Group      
##  Length:12          Min.   : 4.000   Min.   : 4.000   Min.   :1.000  
##  Class :character   1st Qu.: 6.750   1st Qu.: 5.000   1st Qu.:1.000  
##  Mode  :character   Median : 9.500   Median : 6.500   Median :2.000  
##                     Mean   : 9.417   Mean   : 6.417   Mean   :1.583  
##                     3rd Qu.:12.250   3rd Qu.: 7.000   3rd Qu.:2.000  
##                     Max.   :15.000   Max.   :10.000   Max.   :2.000

3.3.2 グループ別での演算の実行

aggregate()関数の機能は、ベクトルデータに対して、グループ分けを行った上で、それぞれのグループのデータに対して設定した関数演算を実行するというものである。 第1引数には、演算対象データの列名~グループ分けデータの列名という書式で、それぞれのデータを指定する。第2引数には演算を行いたいデータが入っているデータフレームを与える。第3引数には、実行したい演算関数の名前を与える。入力するものはあくまで関数の名前だけでよく、()はつける必要はない。

結果はデータフレームで出力される。

#グループ分け変数の設定
Shohoku$Group <- as.factor(c("レギュラー", "レギュラー", "レギュラー", "レギュラー", "レギュラー", "控え", "控え", "控え", "控え", "控え", "控え", "控え"))

#平均を算出
res.mean<-aggregate(score~Group, Shohoku,mean)
print(res.mean)
##        Group    score
## 1 レギュラー 7.400000
## 2       控え 5.714286
#標準偏差を算出
res.sd<-aggregate(score~Group, Shohoku, sd)
print(res.sd)
##        Group    score
## 1 レギュラー 2.302173
## 2       控え 1.112697

4 基本的なデータの可視化

平均や分散、標準偏差、あるいは四分位数といった基本的な統計量の算出方法が分かったので、ここでggplot2というパッケージを用いたデータの可視化の方法の基礎について学習する。

4.1 ggplot2のインストール

ggplot2は美しい描画をしてくれる様々な関数を提供する外部パッケージである。利用するためには、まずTools -> install.packagesを選択し、tidyverseと入力する。ついで、スクリプトの冒頭にlibrary(tidyvers)と書き入れ、パッケージを読み込む。

ggplot2自体は「ggplot2」という名前のパッケージであり、インストールするパッケージも本来であれば「ggplot2」パッケージで良いのだが、「tidyverse」パッケージではggplot2を含め様々な便利なパッケージがまとめられている。後々それらのパッケージも利用していきたいので、ここではtidyverseパッケージを利用する。

4.2 ggplot2をつかった描画の基本的な流れ

Step1:データを与えて描画キャンバスを作る

Step2:与えたデータに対して各種の描画命令を与える

Step3:必要ならば凡例や軸の書式の設定をする

Step4:plot()関数を実行する。

以下では具体的な例を示すためにRに最初から入っているデモ用データのirisを用いる。 irisデータはアイリス(菖蒲)の花びらとがくの長さと幅の実測データが3つの種類(Setosa, vergicolor, virginica)ごとに50点ずつ格納されている。 内容を確認するために、summary()関数とhead()関数(データフレームの冒頭データを出力させる関数)を実行してみた。

d <- iris #irisデータを呼び出してdというオブジェクトに格納
summary(d)
##   Sepal.Length    Sepal.Width     Petal.Length    Petal.Width   
##  Min.   :4.300   Min.   :2.000   Min.   :1.000   Min.   :0.100  
##  1st Qu.:5.100   1st Qu.:2.800   1st Qu.:1.600   1st Qu.:0.300  
##  Median :5.800   Median :3.000   Median :4.350   Median :1.300  
##  Mean   :5.843   Mean   :3.057   Mean   :3.758   Mean   :1.199  
##  3rd Qu.:6.400   3rd Qu.:3.300   3rd Qu.:5.100   3rd Qu.:1.800  
##  Max.   :7.900   Max.   :4.400   Max.   :6.900   Max.   :2.500  
##        Species  
##  setosa    :50  
##  versicolor:50  
##  virginica :50  
##                 
##                 
## 
head(d)
##   Sepal.Length Sepal.Width Petal.Length Petal.Width Species
## 1          5.1         3.5          1.4         0.2  setosa
## 2          4.9         3.0          1.4         0.2  setosa
## 3          4.7         3.2          1.3         0.2  setosa
## 4          4.6         3.1          1.5         0.2  setosa
## 5          5.0         3.6          1.4         0.2  setosa
## 6          5.4         3.9          1.7         0.4  setosa

列名が英語のままだと分かりにくいので、日本語に翻訳した名前に置き換える(->参考)。以下はそのためのコード。

colnames(d) <-  c("がく長","がく幅","花びら長","花びら幅","種類")

4.3 キャンバスの作成

キャンバスの作成にはggplot()関数を用いる。引数にはデータを与える。

g <- ggplot(d)

これでキャンバスのオブジェクトがgという名前で格納された。このgに対して、色々な描画関数を与えていき、最後にgplot()関数で実際に描画させる、というのが基本的な描画のフローとなる。

ちなみに現時点でplot()を実行すると以下のようになる。

plot(g)

このとおり、まだ中身がなにもないので描画エリアだけの描画となる。

4.4 描画関数の実行

4.4.1 箱ひげ図

まずはがく長の箱ひげ図を書かせることにしよう。

4.4.1.1 単純な箱ひげ図

箱ひげ図を書かせる命令はgeom_boxplot()関数となる。この関数の第1引数にaes()関数をあたえる。aes()関数ではさらにy軸のデータを設定する。

g <- ggplot(d) + geom_boxplot(aes(y=がく長))
plot(g)

y をxに返ると以下の通りとなる。

g <- ggplot(d) + geom_boxplot(aes(x=がく長))
plot(g)

4.4.1.2 種類ごとの箱ひげ図

上記はxないしはyのいずれかの軸しか設定していないが、例えば、x軸を種類に設定して、種類ごとの箱ひげ図を書かせたいといった場合には以下の通りにする。

g <- ggplot(d) + geom_boxplot(aes(x=種類, y=がく長))
plot(g)

このように箱ひげ図が種類ごとに描かれる。

4.4.1.3 箱ひげ図へ塗色

箱ひげ図に色を塗りたい場合には、aes()fill(塗りつぶし)の設定を与える。塗分けは箱ひげ図の場合にはグループ単位で行いたいであろう。そういう場合には、グループ分けに使っている変数、すなわち種類を与えると良い。

g <- ggplot(d) + geom_boxplot(aes(x=種類, y=がく長, fill=種類))
plot(g)

4.4.1.4 箱の幅を変える

geom_boxplot()の第2引数(aes()関数の外の引数)としてwidthを与える。デフォルトは0.75に設定されてる。

g <- ggplot(d) + geom_boxplot(aes(x=種類, y=がく長, fill=種類),width=0.3 )
plot(g)

g <- ggplot(d) + geom_boxplot(aes(x=種類, y=がく長, fill=種類),width=1.0 )
plot(g)

g <- ggplot(d) + geom_boxplot(aes(x=種類, y=がく長, fill=種類),width=0.75)
plot(g)

4.4.2 データを点でプロットする

geom_jitter()を使う。引数としてaes()を使って同じようにx、yを設定する。

g <- ggplot(d)+geom_jitter(aes(x=種類, y=がく長))
plot(g)

y軸(がく長)については、それぞれ値を持っているが、x軸については3種類のグループ分けしかない。それをそのまま描画したのでは、同じがく長になっているものが全部重なってしまうので、そうならないように、x軸方向にランダムにバラつかせて描画されている。このx軸方向のバラつきは描画のたびにランダムに変わる。

もう一度描画させると、先ほどと違うばらつき方になる。

g <- ggplot(d)+geom_jitter(aes(x=種類, y=がく長))
plot(g)

4.4.2.1 色やプロット範囲の幅を変える

箱ひげ図の時と同様に、色や幅を変えることができる。色についてはfill(塗りつぶし)ではなくcolorを設定したり、Widthを設定したりして、色や幅を変えることができる。

g <- ggplot(d)+geom_jitter(aes(x=種類, y=がく長, color=種類))
plot(g)

g <- ggplot(d)+geom_jitter(aes(x=種類, y=がく長, color=種類), width = 0.3)
plot(g)

4.4.2.2 Jitterではなく度数分布にする

先の例では点をプロットするときに横方向にランダム化させるjitter表示にしていたが、度数分布にすることもできる。 度数分布の場合には、geom_jitter()の代わりにgeom_dotplot()関数を用いる。引数がやや多いので以下に表で示す。

引数名 内容
aes(x=, y=) x軸、y軸の設定
binaxis どちらの軸の変数についての度数分布にするのか
dotsize プロットする点のサイズ
stackdir どの方向にプロットしていくか。Up(右向き), down(左向き), center(中心ぞろえ)のいずれか
binwidth 度数分布における区間幅の設定
position position_nudge()関数によって描画開始をそれぞれのグループのx軸の中心からどちら向き(正か負か)にどれだけ離すかの設定する。stackdirをcenterにした場合には中心がずれる

以下に、実際にいくつか例示してみる。

g <- ggplot(d)+
  geom_dotplot(aes(x=種類, y=がく長), binaxis = "y", dotsize = 1.0, stackdir = "down",
               binwidth = 0.1, position = position_nudge(-0.025))
plot(g)

g <- ggplot(d)+
  geom_dotplot(aes(x=種類, y=がく長), binaxis = "y", dotsize = 0.50, stackdir = "up",
               binwidth = 0.1, position = position_nudge(0.025))
plot(g)

g <- ggplot(d)+
  geom_dotplot(aes(x=種類, y=がく長), binaxis = "y", dotsize = 0.3, stackdir = "center",
               binwidth = 0.3, position = position_nudge(0.225))
plot(g)

以下はbinaxisをxに設定しているためエラーとなる(xはグループ分け変数であり、概念上度数分布というものが存在しないので)。

g <- ggplot(d)+
  geom_dotplot(aes(x=種類, y=がく長), binaxis = "x", dotsize = 0.3, stackdir = "up",
               binwidth = 0.1, position = position_nudge(0.025))
plot(g)
## Warning: The following aesthetics were dropped during statistical transformation: y
## ℹ This can happen when ggplot fails to infer the correct grouping structure in
##   the data.
## ℹ Did you forget to specify a `group` aesthetic or to convert a numerical
##   variable into a factor?
## Error in `geom_dotplot()`:
## ! Problem while setting up geom.
## ℹ Error occurred in the 1st layer.
## Caused by error in `compute_geom_1()`:
## ! `geom_dotplot()` requires the following missing aesthetics: y

4.4.3 箱ひげ図と点プロットを重ねる

点のプロットと箱ひげ図を重ねることもできる。以下のようにgeom_boxplot()geom_dotplot()geom_jitter()+でつなげればよい。

g<- ggplot(d)+
  geom_boxplot(aes(x=種類,y=がく長))+
  geom_jitter(aes(x=種類, y=がく長, color=種類))
plot(g)

g<- ggplot(d)+
  geom_boxplot(aes(x=種類,y=がく長), width=0.5)+
  geom_dotplot(aes(x=種類, y=がく長, color=種類, fill=種類), binaxis = "y", 
               dotsize = 0.75, stackdir = "down",binwidth = 0.1, 
               position = position_nudge(-0.025))
plot(g)

aes()(エステティック)関数の設定項目はx、y軸とfill(塗りつぶし), color(線や点の色設定)が出てきているが、それ以外に以下のようなものがある。

項目 内容
x, y x軸に設定する変数、y軸に設定する変数
color 点や線の色
fill 面の色
alpha 不透明度 (0が透明、1が不透明)
size 点や文字の大きさ、線の太さ
shape 点の形
linetype 線の種類
group 反復試行の折れ線グラフなど、色や形はそのままで切り分けたいときに。
xmin, xmax, ymin, ymax, xend, yend ものによってつかう。

極めて多いエラーとして、各geom()関数を実行していくときに、前の行の末尾に+を付け忘れたり、最後の行の末尾に+をつけたりしてしまうことがよくある。 もし、エラーが出てグラフ描画ができなかった場合には、これらのミスをしていないかを確認してみてほしい。

4.4.4 棒グラフ

平均や標準偏差が得られているなら棒グラフとしてそれらを描きたいであろう。その場合には、まずは平均と標準偏差のグループごとのデータを用意した上で、以下のようにする。

#データの準備
d.mean <- aggregate(がく長~種類,d, mean)
d.sd <- aggregate(がく長~種類,d, sd)
d.mean
##         種類 がく長
## 1     setosa  5.006
## 2 versicolor  5.936
## 3  virginica  6.588
d.sd
##         種類    がく長
## 1     setosa 0.3524897
## 2 versicolor 0.5161711
## 3  virginica 0.6358796
d2 <- data.frame(
  種類 = d.mean$種類,
  平均 = d.mean$がく長,
  標準偏差= d.sd$がく長
)
 print(d2) 
##         種類  平均  標準偏差
## 1     setosa 5.006 0.3524897
## 2 versicolor 5.936 0.5161711
## 3  virginica 6.588 0.6358796
 g <- ggplot(d2)+
  geom_bar(aes(x=種類,y=平均,fill=種類),stat="identity")+
  geom_errorbar(aes(x=種類,y=平均,ymin=平均-標準偏差, ymax=平均+標準偏差,width=0.15))
plot(g)

今まで、キャンバスを作った際にデータフレームを与え、その後、描画をする際にxとyを指摘してきましたが、キャンバスを作るときにxとyも指定することができる。そうすると、描画をするときに、いちいちx、yを指定しなくても良くなる。

g <- ggplot(d2)+
  geom_bar(aes(x=種類,y=平均,fill=種類),stat="identity")+
  geom_errorbar(aes(x=種類,y=平均,ymin=平均-標準偏差, ymax=平均+標準偏差,width=0.3))
plot(g)

g <- ggplot(d2,aes(x=種類,y=平均))+
  geom_bar(aes(fill=種類),stat="identity")+
  geom_errorbar(aes(ymin=平均-標準偏差, ymax=平均+標準偏差,width=0.3))
plot(g)

4.4.5 ヒストグラム

ヒストグラムを作成する方法は2つある。

  1. geom_bar()statcountに設定する。
  2. geom_histgram()を使う

4.4.5.1 geom_bar()を使う

度数分布をグラフ表示したものをヒストグラムと呼ぶ。ヒストグラムは棒グラフの派生形として、以下のように、statcountに設定することで描くことができる。ただし、この場合には区間幅を自分で設定することはできない。

g <- ggplot(d)+
  geom_bar(aes(x=がく長),stat="count")
plot(g)

4.4.5.2 geom_histgram()を使う

geom_histgramを使う場合、区間幅はデフォルトではデータの範囲を30等分した値が自動的に設定されるが、binwidthを設定することによって任意の区間幅に設定することもできる。

g <- ggplot(d)+
  geom_histogram(aes(x=がく長), binwidth = 0.1)
plot(g)

これだと区間の境界がわかりにくいので、境界を描かせるためには以下のようにする。

g <- ggplot(d)+
  geom_histogram(aes(x=がく長),color="white", binwidth = 0.1)
plot(g)

4.4.5.3 種類ごとの色分け

種類ごとに分けて分布表を作る場合は、geom_barにせよ、geom_histgramにせよaes()の中でfillを設定する。

g <- ggplot(d)+
  geom_bar(aes(x=がく長, fill=種類),stat="count")
plot(g)

g <- ggplot(d)+
  geom_histogram(aes(x=がく長, fill=種類),color="white", binwidth = 0.1)
plot(g)

なお、例えば、がく長が5.7は度数として8となっているが、これは3つの種類をすべて積み上げたものである。すなわち、setosaにおいてがく長5.8なのは2点、Versicolorでは5点、virginicaで1点となっている。

4.4.5.4 種類別のヒストグラム作成

あくまで種類ごとにそれぞれ別々のヒストグラムにしたい、ということであれば、以下のように2つの方法が考えられる。

方法1 データを端からバラバラにして3つのグラフを別々に描く

この場合には、データをfilter()関数(tidyverseに含まれる関数)を用いて種類ごとに分け、それぞれを描画する。ただし、この場合、coord_cartesian(xlim =)関数を使ってx軸の変域を設定しないと、3つのグラフのx軸の変域がバラバラになってしまう。

g <- ggplot(filter(d,種類=="setosa"))+
  geom_histogram(aes(x=がく長),color="white", binwidth = 0.1)+
  coord_cartesian(xlim = c(4, 8))
plot(g)

g <- ggplot(filter(d,種類=="versicolor"))+
  geom_histogram(aes(x=がく長),color="white", binwidth = 0.1)+
  coord_cartesian(xlim = c(4, 8))
plot(g)

g <- ggplot(filter(d,種類=="virginica") )+
  geom_histogram(aes(x=がく長),color="white", binwidth = 0.1)+
  coord_cartesian(xlim = c(4, 8))
plot(g)

変数の内容を知るための関数としてstr()という関数がある。実際にこれをfilter()関数の結果に対してかけてみると、以下の通りとなる。

res<-filter(d,種類=="setosa")
str(res)
## 'data.frame':    50 obs. of  5 variables:
##  $ がく長  : num  5.1 4.9 4.7 4.6 5 5.4 4.6 5 4.4 4.9 ...
##  $ がく幅  : num  3.5 3 3.2 3.1 3.6 3.9 3.4 3.4 2.9 3.1 ...
##  $ 花びら長: num  1.4 1.4 1.3 1.5 1.4 1.7 1.4 1.5 1.4 1.5 ...
##  $ 花びら幅: num  0.2 0.2 0.2 0.2 0.2 0.4 0.3 0.2 0.2 0.1 ...
##  $ 種類    : Factor w/ 3 levels "setosa","versicolor",..: 1 1 1 1 1 1 1 1 1 1 ...

データ点数は50点とあり、確かにsetosaのみを取り出せているのがわかる。一方で「種類」列の情報としては、この列は要因型であり、3つの水準(setosa, versicolorだけが表示されているが・・・)で構成されていることが記載されていることがわかる。

方法2: 1つの絵の中で3つに分割して描く(facet_wrap()の利用)

facet_wrap()を利用することで1枚のキャンバスを分割して表示させることができる。

g <- ggplot(d )+
  geom_histogram(aes(x=がく長),color="white", binwidth = 0.1)+
    facet_wrap(d$種類) 
plot(g)

デフォルトでは、横方向(列方向)に分割されていくが、nrow(行数)を設定することによって縦方向(行方向)にも分割させることができる。

g <- ggplot(d )+
  geom_histogram(aes(x=がく長),color="white", binwidth = 0.1)+
    facet_wrap(d$種類,nrow=2) 
plot(g)

4.4.6 散布図

2つのデータの関係性を可視化するグラフの代表格は散布図である。散布図は以下のようにして描く。

g <- ggplot(d)+
  geom_point(aes(x=がく長, y=花びら長))
plot(g)

4.4.6.1 種類ごとに色分け

geom_point()aes()colorを与える。

g <- ggplot(d)+
  geom_point(aes(x=がく長, y=花びら長, color=種類))
plot(g)

4.4.6.2 種類ごとにプロットの形を分ける

geom_point()aes()shapeを与える。

g <- ggplot(d)+
  geom_point(aes(x=がく長, y=花びら長, color=種類, shape=種類))
plot(g)

分かりやすく、がく長と花びら長に関係性として、setosaにはほぼ関係性はないが、versicolorとvirginicaにはがく長が長くなると、花びら長も長くなる傾向にあることが分かった。

散布図は回帰分析を行う際の結果の可視化でも改めて触れることにする。

4.5 書式や凡例を弄る

4.5.1 軸のタイトルを消したり書き換えたり

デフォルトでは軸タイトルはaes()で設定した各軸の変数名がそのまま入るが、xlabylabを使って自分の好きなように変えることができる。 引数にNULLを設定すると軸タイトルを消せる。

g <- ggplot(d2,aes(x=種類,y=平均))+
  geom_bar(aes(fill=種類),stat="identity")+
  geom_errorbar(aes(ymin=平均-標準偏差, ymax=平均+標準偏差,width=0.3))+
  xlab(NULL)+ylab("がく長")
plot(g)

4.5.2 凡例タイトルを書き換える

labs()を使う。 引数には、aes()で色を指定するときに使った引数に、書き換えたいタイトルを与える。 以下の例では、fillを使って色を与えているので、fill=****というのを引数としてlabelに与える。

g <- ggplot(d2,aes(x=種類,y=平均))+
  geom_bar(aes(fill=種類),stat="identity")+
  geom_errorbar(aes(ymin=平均-標準偏差, ymax=平均+標準偏差,width=0.3))+
  xlab(NULL)+ylab("学長")+labs(fill="種類")
plot(g)

ちなみにlabsは引数として以下のものを取る。

項目 内容
title タイトル
subtitle サブタイトル
x x軸ラベル
y y軸ラベル
caption キャプション
fill またはcolor 凡例ラベルのタイトル

この通りに、xlabやylabもlabsでまとめて設定することもできる。

g <- ggplot(d2,aes(x=種類,y=平均))+
  geom_bar(aes(fill=種類),stat="identity")+
  geom_errorbar(aes(ymin=平均-標準偏差, ymax=平均+標準偏差,width=0.3))+
  labs(x=NULL, y="がく長", fill="種類")
plot(g)

4.5.3 凡例を消す

theme(legend.position = "none")を使う。

g <- ggplot(d2,aes(x=種類,y=平均))+
  geom_bar(aes(fill=種類),stat="identity")+
  geom_errorbar(aes(ymin=平均-標準偏差, ymax=平均+標準偏差,width=0.3))+
  xlab(NULL)+ylab("がく長")+
  theme(legend.position = "none")
plot(g)

4.5.4 軸タイトルや軸ラベル、凡例の文字サイズを換える

凡例を消した時と同じようにthemeを使う。引数には以下のようなものを入れる。 axis.text.x=element_text(size=15) axis.text.y=element_text(size=15) axis.title.x=element_text(size=15) axis.title.y=element_text(size=15) legend.text=element_text(size=9)

g <- ggplot(d2,aes(x=種類,y=平均))+
  geom_bar(aes(fill=種類),stat="identity")+
  geom_errorbar(aes(ymin=平均-標準偏差, ymax=平均+標準偏差,width=0.3))+
  xlab("種類")+ylab("花弁の幅")+
  theme(legend.position = 'none')+
  theme(
    axis.title.x=element_text(size=15),
    axis.title.y=element_text(size=15),
    axis.text.x=element_text(size=10),
    axis.text.y=element_text(size=10),
    legend.text=element_text(size=5),
    legend.title = element_text(size=8)  )

plot(g)

themeの設定項目

themeは細かくいろいろ設定できる。設定できるものは以下の通り。

名前 説明 要素の型
text すべてのテキスト要素 element_text()
axis.title 両軸ラベルの体裁 element_text()
axis.title.x x軸ラベルの体裁 element_text()
axis.title.y y軸ラベルの体裁 element_text()
axis.text 両軸目盛ラベルの体裁 element_text()
axis.text.x x軸目盛ラベルの体裁 element_text()
axis.text.y y軸目盛ラベルの体裁 element_text()
legend.text 凡例項目の体裁 element_text()
legend.title 凡例タイトルの体裁 element_text()
plot.title タイトルの体裁 element_text()
strip.text 両方向ファセットラベルテキストの体裁 element_text()
strip.text.x 水平方向ファセットラベルテキストの体裁 element_text()
strip.text.y 垂直方向ファセットラベルテキストの体裁 element_text()
legend.position 凡例の位置 left, right, bottom, top, c(x,y) :x,yは0~1
legend.background 凡例の背景 element_rect(fill=“xxx”, color=“yyy”) fillは塗りつぶし、colorは枠線
plot.background プロット全体の背景 element_rect(fill=“xxx”, color=“yyy”)
panel.background プロット領域の背景 element_rect()
panel.border プロット領域の枠線 element_rect(linetype=“xxx”)
rect 全ての長方形要素 element_rect()
panel.grid.major 主目盛線 element_line()
panel.grid.major.x 主目盛線の垂直方向 element_line()
panel.grid.major.y 主目盛線の水平方向 element_line()
panel.grid.minor 補助目盛線 element_line()
panel.grid.minor.x 補助目盛線の垂直方向 element_line()
panel.grid.minor.y 補助目盛線の水平方向 element_line()
axis.line 軸に沿った線 element_line()
line すべての線要素 element_line()

引用(順序は改変しました):https://knknkn.hatenablog.com/entry/2019/02/23/181311

element_text()の設定項目

設定項目 element_text()の引数
角度 angle
水平位置 hjust(0:左揃え~1:右揃え)
垂直位置 vjust(0:下揃え~1:上揃え)
サイズ size
colour
スタイル face(bold or italic)
フォントファミリー family

参考:http://mukkujohn.hatenablog.com/entry/2016/10/11/220722

element_xxxのその他の項目 element_rect(fill, color, size, linetype, inherit.blank) — 長方形

項目 内容
fill 塗りつぶしの色
color 枠の色

element_line(color, size, linetype, lineend, arrow, inherit.blank)— 線

element_blank() — 空 消したい要素にはこれを指定する

参考:https://heavywatal.github.io/rstats/ggplot2.html

4.5.5 目盛ラベルを変える

目盛ラベルを変えるには、scale_x_****scale_y_***を使う。 以下は、x軸の目盛りをsetosa, versicolor, virgnicaから、a, b, cに変え、y軸を普通の目盛から対数目盛に変えた例である。

g <- ggplot(d)+
  geom_bar(aes(x=種類,fill=種類),stat="count") +
  scale_x_discrete(labels=c("a","b","c"))+
  scale_y_log10(limits=c(1,1000))
plot(g)

***の部分には、いずれも以下のものが入る。

*** 内容
discrete 離散目盛
continuous 連続目盛
log10 対数目盛
reverse 目盛の逆転
sqrt 平方根目盛

また、代表的な引数としては以下のものが入る。

引数 内容
labels 離散目盛のラベルを変える(以下のbreaksを併用すると、連続変数でも可能)
breaks 連続変数の目盛のうち指定したものだけを表示させる
limits 連続変数の表示範囲を指定したものにする.xlim,ylimと同じ効果。
name 変数の軸ラベルを変える。xlab,ylabと同じ効果。

結局、軸タイトルを変える方法って、色々あるということである。 xlabでもよいし、labs(x=)dでもよいし、scale_x_***(name=)でも変えれる。

4.5.6 凡例の各項目のラベルを書き換える。

グループ分けにfillを使っている場合は、sacle_fill_dicrete()を、Colorを使っている場合はscale_color_discrete()を使う。

g <- ggplot(d)+
  geom_bar(aes(x=種類,fill=種類),stat="count")+
  scale_fill_discrete(label=c("花1","花2","花3"))
plot(g)

5 データハンドリング(基礎)

5.1 抽出

すでにこちらで紹介したものであるが、改めてここで紹介する。。 グループ分け変数を持ったデータを処理するときには、ある特定のグループのデータだけを抜き出したいということはよくある そうした「データの抽出」を行うための関数としてtidyverseパッケージにはfilter()関数が設けられている。

利用法として、第1引数に元のデータを与え、第2引数に検索条件を与える。検索条件の書式のパターンは以下の通り。

書式 動作
元データ$列名==Key 列名で指定した列の値がKeyと一致しているものを抽出。「=」を重ねる点がポイント。
元データ$列名>Key 列名で指定した列の値がKeyよりも大きいものを抽出
元データ$列名>=Key 列名で指定した列の値がKeyと等しいかKeyよりも大きいものを抽出
元データ$列名<Key 列名で指定した列の値がKeyよりも小さいものを抽出
元データ$列名<=Key 列名で指定した列の値がKeyと等しいかKeyよりも小さいものを抽出
summary(d)
##      がく長          がく幅         花びら長        花びら幅    
##  Min.   :4.300   Min.   :2.000   Min.   :1.000   Min.   :0.100  
##  1st Qu.:5.100   1st Qu.:2.800   1st Qu.:1.600   1st Qu.:0.300  
##  Median :5.800   Median :3.000   Median :4.350   Median :1.300  
##  Mean   :5.843   Mean   :3.057   Mean   :3.758   Mean   :1.199  
##  3rd Qu.:6.400   3rd Qu.:3.300   3rd Qu.:5.100   3rd Qu.:1.800  
##  Max.   :7.900   Max.   :4.400   Max.   :6.900   Max.   :2.500  
##          種類   
##  setosa    :50  
##  versicolor:50  
##  virginica :50  
##                 
##                 
## 
res<- filter(d, d$種類=="virginica")
summary(res)
##      がく長          がく幅         花びら長        花びら幅    
##  Min.   :4.900   Min.   :2.200   Min.   :4.500   Min.   :1.400  
##  1st Qu.:6.225   1st Qu.:2.800   1st Qu.:5.100   1st Qu.:1.800  
##  Median :6.500   Median :3.000   Median :5.550   Median :2.000  
##  Mean   :6.588   Mean   :2.974   Mean   :5.552   Mean   :2.026  
##  3rd Qu.:6.900   3rd Qu.:3.175   3rd Qu.:5.875   3rd Qu.:2.300  
##  Max.   :7.900   Max.   :3.800   Max.   :6.900   Max.   :2.500  
##          種類   
##  setosa    : 0  
##  versicolor: 0  
##  virginica :50  
##                 
##                 
## 
res <- filter(d, d$がく長<=5)
summary(res)
##      がく長          がく幅         花びら長        花びら幅     
##  Min.   :4.300   Min.   :2.000   Min.   :1.000   Min.   :0.1000  
##  1st Qu.:4.600   1st Qu.:3.000   1st Qu.:1.375   1st Qu.:0.2000  
##  Median :4.850   Median :3.150   Median :1.400   Median :0.2000  
##  Mean   :4.787   Mean   :3.091   Mean   :1.703   Mean   :0.3406  
##  3rd Qu.:5.000   3rd Qu.:3.400   3rd Qu.:1.600   3rd Qu.:0.3000  
##  Max.   :5.000   Max.   :3.600   Max.   :4.500   Max.   :1.7000  
##          種類   
##  setosa    :28  
##  versicolor: 3  
##  virginica : 1  
##                 
##                 
## 
res <- filter(d, d$がく長>=6)
summary(res)
##      がく長         がく幅         花びら長        花びら幅             種類   
##  Min.   :6.00   Min.   :2.200   Min.   :4.000   Min.   :1.000   setosa    : 0  
##  1st Qu.:6.30   1st Qu.:2.800   1st Qu.:4.700   1st Qu.:1.500   versicolor:24  
##  Median :6.50   Median :3.000   Median :5.200   Median :1.800   virginica :43  
##  Mean   :6.61   Mean   :2.966   Mean   :5.263   Mean   :1.816                  
##  3rd Qu.:6.85   3rd Qu.:3.150   3rd Qu.:5.700   3rd Qu.:2.100                  
##  Max.   :7.90   Max.   :3.800   Max.   :6.900   Max.   :2.500

条件は複数並べることもできる。|は複数の条件のいずれかにマッチすれば抽出される。&は複数条件の全てにマッチするものが抽出される。 それぞれ以下に示す。

res <- filter(d, d$がく長>=6 | d$花びら長>1.5)
summary(res)
##      がく長          がく幅        花びら長        花びら幅             種類   
##  Min.   :4.700   Min.   :2.00   Min.   :1.600   Min.   :0.200   setosa    :13  
##  1st Qu.:5.600   1st Qu.:2.70   1st Qu.:4.000   1st Qu.:1.300   versicolor:50  
##  Median :6.100   Median :3.00   Median :4.700   Median :1.500   virginica :50  
##  Mean   :6.125   Mean   :2.94   Mean   :4.535   Mean   :1.519                  
##  3rd Qu.:6.600   3rd Qu.:3.20   3rd Qu.:5.400   3rd Qu.:2.000                  
##  Max.   :7.900   Max.   :3.90   Max.   :6.900   Max.   :2.500
res <- filter(d, d$がく長>=6 & d$花びら長>1.5)
summary(res)
##      がく長         がく幅         花びら長        花びら幅             種類   
##  Min.   :6.00   Min.   :2.200   Min.   :4.000   Min.   :1.000   setosa    : 0  
##  1st Qu.:6.30   1st Qu.:2.800   1st Qu.:4.700   1st Qu.:1.500   versicolor:24  
##  Median :6.50   Median :3.000   Median :5.200   Median :1.800   virginica :43  
##  Mean   :6.61   Mean   :2.966   Mean   :5.263   Mean   :1.816                  
##  3rd Qu.:6.85   3rd Qu.:3.150   3rd Qu.:5.700   3rd Qu.:2.100                  
##  Max.   :7.90   Max.   :3.800   Max.   :6.900   Max.   :2.500
res <- filter(d, d$がく長>=5 & d$がく長<6.5) # 5<=がく長<6.5
summary(res)
##      がく長          がく幅         花びら長        花びら幅    
##  Min.   :5.000   Min.   :2.000   Min.   :1.200   Min.   :0.100  
##  1st Qu.:5.300   1st Qu.:2.700   1st Qu.:1.600   1st Qu.:0.400  
##  Median :5.700   Median :3.000   Median :4.100   Median :1.300  
##  Mean   :5.691   Mean   :3.047   Mean   :3.577   Mean   :1.133  
##  3rd Qu.:6.100   3rd Qu.:3.400   3rd Qu.:4.800   3rd Qu.:1.600  
##  Max.   :6.400   Max.   :4.400   Max.   :6.000   Max.   :2.500  
##          種類   
##  setosa    :30  
##  versicolor:40  
##  virginica :23  
##                 
##                 
## 
res <- filter(d, d$がく長<5 | d$がく長>=6.5) # がく長<5, 6.5<=学長
summary(res)
##      がく長          がく幅         花びら長        花びら幅    
##  Min.   :4.300   Min.   :2.300   Min.   :1.000   Min.   :0.100  
##  1st Qu.:4.800   1st Qu.:3.000   1st Qu.:1.500   1st Qu.:0.200  
##  Median :6.600   Median :3.100   Median :4.800   Median :1.500  
##  Mean   :6.091   Mean   :3.074   Mean   :4.053   Mean   :1.307  
##  3rd Qu.:6.900   3rd Qu.:3.200   3rd Qu.:5.800   3rd Qu.:2.100  
##  Max.   :7.900   Max.   :3.800   Max.   :6.900   Max.   :2.500  
##          種類   
##  setosa    :20  
##  versicolor:10  
##  virginica :27  
##                 
##                 
## 

抽出したデータのある列だけが必要だという場合には、抽出したデータを収めたオブジェクトにに$[]をつけて指定すればよい。

res <- filter(d, d$がく長>=6 | d$花びら長>1.5)
res<- res$花びら長
print(res)
##   [1] 1.7 1.6 1.7 1.7 1.7 1.9 1.6 1.6 1.6 1.6 1.6 1.9 1.6 4.7 4.5 4.9 4.0 4.6
##  [19] 4.5 4.7 3.3 4.6 3.9 3.5 4.2 4.0 4.7 3.6 4.4 4.5 4.1 4.5 3.9 4.8 4.0 4.9
##  [37] 4.7 4.3 4.4 4.8 5.0 4.5 3.5 3.8 3.7 3.9 5.1 4.5 4.5 4.7 4.4 4.1 4.0 4.4
##  [55] 4.6 4.0 3.3 4.2 4.2 4.2 4.3 3.0 4.1 6.0 5.1 5.9 5.6 5.8 6.6 4.5 6.3 5.8
##  [73] 6.1 5.1 5.3 5.5 5.0 5.1 5.3 5.5 6.7 6.9 5.0 5.7 4.9 6.7 4.9 5.7 6.0 4.8
##  [91] 4.9 5.6 5.8 6.1 6.4 5.6 5.1 5.6 6.1 5.6 5.5 4.8 5.4 5.6 5.1 5.1 5.9 5.7
## [109] 5.2 5.0 5.2 5.4 5.1
res<- res["花びら長"]
print(res)
## [1] NA
#次のように関数に対して直接[]や$を書ける
res2 <- filter(d, d$がく長>=6 | d$花びら長>1.5)$花びら長
print(res2)
##   [1] 1.7 1.6 1.7 1.7 1.7 1.9 1.6 1.6 1.6 1.6 1.6 1.9 1.6 4.7 4.5 4.9 4.0 4.6
##  [19] 4.5 4.7 3.3 4.6 3.9 3.5 4.2 4.0 4.7 3.6 4.4 4.5 4.1 4.5 3.9 4.8 4.0 4.9
##  [37] 4.7 4.3 4.4 4.8 5.0 4.5 3.5 3.8 3.7 3.9 5.1 4.5 4.5 4.7 4.4 4.1 4.0 4.4
##  [55] 4.6 4.0 3.3 4.2 4.2 4.2 4.3 3.0 4.1 6.0 5.1 5.9 5.6 5.8 6.6 4.5 6.3 5.8
##  [73] 6.1 5.1 5.3 5.5 5.0 5.1 5.3 5.5 6.7 6.9 5.0 5.7 4.9 6.7 4.9 5.7 6.0 4.8
##  [91] 4.9 5.6 5.8 6.1 6.4 5.6 5.1 5.6 6.1 5.6 5.5 4.8 5.4 5.6 5.1 5.1 5.9 5.7
## [109] 5.2 5.0 5.2 5.4 5.1
#次のように関数に対して直接[]や$を書ける
res2 <- filter(d, d$がく長>=6 | d$花びら長>1.5)["花びら長"]
print(res2)
##     花びら長
## 1        1.7
## 2        1.6
## 3        1.7
## 4        1.7
## 5        1.7
## 6        1.9
## 7        1.6
## 8        1.6
## 9        1.6
## 10       1.6
## 11       1.6
## 12       1.9
## 13       1.6
## 14       4.7
## 15       4.5
## 16       4.9
## 17       4.0
## 18       4.6
## 19       4.5
## 20       4.7
## 21       3.3
## 22       4.6
## 23       3.9
## 24       3.5
## 25       4.2
## 26       4.0
## 27       4.7
## 28       3.6
## 29       4.4
## 30       4.5
## 31       4.1
## 32       4.5
## 33       3.9
## 34       4.8
## 35       4.0
## 36       4.9
## 37       4.7
## 38       4.3
## 39       4.4
## 40       4.8
## 41       5.0
## 42       4.5
## 43       3.5
## 44       3.8
## 45       3.7
## 46       3.9
## 47       5.1
## 48       4.5
## 49       4.5
## 50       4.7
## 51       4.4
## 52       4.1
## 53       4.0
## 54       4.4
## 55       4.6
## 56       4.0
## 57       3.3
## 58       4.2
## 59       4.2
## 60       4.2
## 61       4.3
## 62       3.0
## 63       4.1
## 64       6.0
## 65       5.1
## 66       5.9
## 67       5.6
## 68       5.8
## 69       6.6
## 70       4.5
## 71       6.3
## 72       5.8
## 73       6.1
## 74       5.1
## 75       5.3
## 76       5.5
## 77       5.0
## 78       5.1
## 79       5.3
## 80       5.5
## 81       6.7
## 82       6.9
## 83       5.0
## 84       5.7
## 85       4.9
## 86       6.7
## 87       4.9
## 88       5.7
## 89       6.0
## 90       4.8
## 91       4.9
## 92       5.6
## 93       5.8
## 94       6.1
## 95       6.4
## 96       5.6
## 97       5.1
## 98       5.6
## 99       6.1
## 100      5.6
## 101      5.5
## 102      4.8
## 103      5.4
## 104      5.6
## 105      5.1
## 106      5.1
## 107      5.9
## 108      5.7
## 109      5.2
## 110      5.0
## 111      5.2
## 112      5.4
## 113      5.1
#複数の列を取り出す場合には[c()]を使う
res3 <- filter(d, d$がく長>=6 | d$花びら長>1.5)[c("花びら長","花びら幅")]
print(res3)
##     花びら長 花びら幅
## 1        1.7      0.4
## 2        1.6      0.2
## 3        1.7      0.3
## 4        1.7      0.2
## 5        1.7      0.5
## 6        1.9      0.2
## 7        1.6      0.2
## 8        1.6      0.4
## 9        1.6      0.2
## 10       1.6      0.2
## 11       1.6      0.6
## 12       1.9      0.4
## 13       1.6      0.2
## 14       4.7      1.4
## 15       4.5      1.5
## 16       4.9      1.5
## 17       4.0      1.3
## 18       4.6      1.5
## 19       4.5      1.3
## 20       4.7      1.6
## 21       3.3      1.0
## 22       4.6      1.3
## 23       3.9      1.4
## 24       3.5      1.0
## 25       4.2      1.5
## 26       4.0      1.0
## 27       4.7      1.4
## 28       3.6      1.3
## 29       4.4      1.4
## 30       4.5      1.5
## 31       4.1      1.0
## 32       4.5      1.5
## 33       3.9      1.1
## 34       4.8      1.8
## 35       4.0      1.3
## 36       4.9      1.5
## 37       4.7      1.2
## 38       4.3      1.3
## 39       4.4      1.4
## 40       4.8      1.4
## 41       5.0      1.7
## 42       4.5      1.5
## 43       3.5      1.0
## 44       3.8      1.1
## 45       3.7      1.0
## 46       3.9      1.2
## 47       5.1      1.6
## 48       4.5      1.5
## 49       4.5      1.6
## 50       4.7      1.5
## 51       4.4      1.3
## 52       4.1      1.3
## 53       4.0      1.3
## 54       4.4      1.2
## 55       4.6      1.4
## 56       4.0      1.2
## 57       3.3      1.0
## 58       4.2      1.3
## 59       4.2      1.2
## 60       4.2      1.3
## 61       4.3      1.3
## 62       3.0      1.1
## 63       4.1      1.3
## 64       6.0      2.5
## 65       5.1      1.9
## 66       5.9      2.1
## 67       5.6      1.8
## 68       5.8      2.2
## 69       6.6      2.1
## 70       4.5      1.7
## 71       6.3      1.8
## 72       5.8      1.8
## 73       6.1      2.5
## 74       5.1      2.0
## 75       5.3      1.9
## 76       5.5      2.1
## 77       5.0      2.0
## 78       5.1      2.4
## 79       5.3      2.3
## 80       5.5      1.8
## 81       6.7      2.2
## 82       6.9      2.3
## 83       5.0      1.5
## 84       5.7      2.3
## 85       4.9      2.0
## 86       6.7      2.0
## 87       4.9      1.8
## 88       5.7      2.1
## 89       6.0      1.8
## 90       4.8      1.8
## 91       4.9      1.8
## 92       5.6      2.1
## 93       5.8      1.6
## 94       6.1      1.9
## 95       6.4      2.0
## 96       5.6      2.2
## 97       5.1      1.5
## 98       5.6      1.4
## 99       6.1      2.3
## 100      5.6      2.4
## 101      5.5      1.8
## 102      4.8      1.8
## 103      5.4      2.1
## 104      5.6      2.4
## 105      5.1      2.3
## 106      5.1      1.9
## 107      5.9      2.3
## 108      5.7      2.5
## 109      5.2      2.3
## 110      5.0      1.9
## 111      5.2      2.0
## 112      5.4      2.3
## 113      5.1      1.8

5.2 数値の置換

数値の置換にはifelse()関数を応用する。ifelse()関数は、第1引数に条件を記述し(条件の記述方法は検索と同じ)、第2引数には条件にマッチした場合に置き換える値、第3引数にはマッチしなかった場合に置き換える値を記載する。ifelse()関数の実行結果として最終的に返ってくるのは、そうした置き換えが行われたベクトルとなる。あくまでベクトルが返ってくるだけなので、実際に置き換えをデータに反映させるためには、元のデータの元の列に返ってきたベクトルを代入するなり、元のデータに新しい列を作ってベクトルを代入するなりする必要がある。

「値」と記載しているが、コンピュータの用語として「値」という場合には、数値だけでなく文字列も含められていることは覚えておこう。数値だけをいう場合には「数値」、文字列だけを言う場合には「文字列」とはっきりと言わなければならない。

以下の例ではがく長が平均値より上か下かで平均以上なら1、未満なら0を割り当てる変数を作っている。

d2 <- d 
d2$bがく長 <- ifelse(d2$がく長 >= mean(d2$がく長), 1, 0)
summary(d2)
##      がく長          がく幅         花びら長        花びら幅    
##  Min.   :4.300   Min.   :2.000   Min.   :1.000   Min.   :0.100  
##  1st Qu.:5.100   1st Qu.:2.800   1st Qu.:1.600   1st Qu.:0.300  
##  Median :5.800   Median :3.000   Median :4.350   Median :1.300  
##  Mean   :5.843   Mean   :3.057   Mean   :3.758   Mean   :1.199  
##  3rd Qu.:6.400   3rd Qu.:3.300   3rd Qu.:5.100   3rd Qu.:1.800  
##  Max.   :7.900   Max.   :4.400   Max.   :6.900   Max.   :2.500  
##          種類       bがく長      
##  setosa    :50   Min.   :0.0000  
##  versicolor:50   1st Qu.:0.0000  
##  virginica :50   Median :0.0000  
##                  Mean   :0.4667  
##                  3rd Qu.:1.0000  
##                  Max.   :1.0000
d2$bがく長 <- as.factor(d2$bがく長)
summary(d2)
##      がく長          がく幅         花びら長        花びら幅    
##  Min.   :4.300   Min.   :2.000   Min.   :1.000   Min.   :0.100  
##  1st Qu.:5.100   1st Qu.:2.800   1st Qu.:1.600   1st Qu.:0.300  
##  Median :5.800   Median :3.000   Median :4.350   Median :1.300  
##  Mean   :5.843   Mean   :3.057   Mean   :3.758   Mean   :1.199  
##  3rd Qu.:6.400   3rd Qu.:3.300   3rd Qu.:5.100   3rd Qu.:1.800  
##  Max.   :7.900   Max.   :4.400   Max.   :6.900   Max.   :2.500  
##          種類    bがく長
##  setosa    :50   0:80   
##  versicolor:50   1:70   
##  virginica :50          
##                         
##                         
## 

5.2.1 文字列の置換

文字列型の列や要因型の列を置換したい場合にはifelse()関数を使うよりも、recorde()関数を使うほうが簡単である。変換したいものの対応を第2引数に以下の例の通りに与えればよい。変換式が指定されていないものは元の値が維持される。

d2 <- d 
d2$種類 <- recode(d2$種類, "setosa"="セトサ", "versicolor"="バーシカラー")
summary(d2$種類)
##       セトサ バーシカラー    virginica 
##           50           50           50

もちろんifelse()関数を使うこともできるが、列が要因型の場合には、一旦文字列型に変換してからでないと、思ったような結果とならない場合がある。例えば、以下の例ではsetosaだけをセトサに置き換えたいと思って実行したのにもかかわらず、列が要因型ではなく勝手に文字列型にされ、さらに、他のもの種類名も勝手に23に設定されてしまっているのがわかる。

d2 <- d
d2$種類 <- ifelse(d2$種類=="setosa", "セトサ", d2$種類 )
summary(d2$種類) #サマリー表示 classがCharacter(文字列型)となっている
d2$種類 #中身を確認してみる
##    Length     Class      Mode 
##       150 character character 
##   [1] "セトサ" "セトサ" "セトサ" "セトサ" "セトサ" "セトサ" "セトサ" "セトサ"
##   [9] "セトサ" "セトサ" "セトサ" "セトサ" "セトサ" "セトサ" "セトサ" "セトサ"
##  [17] "セトサ" "セトサ" "セトサ" "セトサ" "セトサ" "セトサ" "セトサ" "セトサ"
##  [25] "セトサ" "セトサ" "セトサ" "セトサ" "セトサ" "セトサ" "セトサ" "セトサ"
##  [33] "セトサ" "セトサ" "セトサ" "セトサ" "セトサ" "セトサ" "セトサ" "セトサ"
##  [41] "セトサ" "セトサ" "セトサ" "セトサ" "セトサ" "セトサ" "セトサ" "セトサ"
##  [49] "セトサ" "セトサ" "2"      "2"      "2"      "2"      "2"      "2"     
##  [57] "2"      "2"      "2"      "2"      "2"      "2"      "2"      "2"     
##  [65] "2"      "2"      "2"      "2"      "2"      "2"      "2"      "2"     
##  [73] "2"      "2"      "2"      "2"      "2"      "2"      "2"      "2"     
##  [81] "2"      "2"      "2"      "2"      "2"      "2"      "2"      "2"     
##  [89] "2"      "2"      "2"      "2"      "2"      "2"      "2"      "2"     
##  [97] "2"      "2"      "2"      "2"      "3"      "3"      "3"      "3"     
## [105] "3"      "3"      "3"      "3"      "3"      "3"      "3"      "3"     
## [113] "3"      "3"      "3"      "3"      "3"      "3"      "3"      "3"     
## [121] "3"      "3"      "3"      "3"      "3"      "3"      "3"      "3"     
## [129] "3"      "3"      "3"      "3"      "3"      "3"      "3"      "3"     
## [137] "3"      "3"      "3"      "3"      "3"      "3"      "3"      "3"     
## [145] "3"      "3"      "3"      "3"      "3"      "3"

意図した通りのことをしたいならば、以下のようになる。

d2 <- d
d2$種類 <- as.character(d2$種類) #要因型から文字列型に変換
d2$種類 <- ifelse(d2$種類=="setosa", "セトサ", d2$種類 )
d2$種類 #中身を確認してみる
summary(d2$種類) #サマリー表示
d2$種類 <- as.factor(d2$種類) #文字列型から要因型に変換
summary(d2$種類) #サマリー表示 要因型に変換したので表示が変わる
##   [1] "セトサ"     "セトサ"     "セトサ"     "セトサ"     "セトサ"    
##   [6] "セトサ"     "セトサ"     "セトサ"     "セトサ"     "セトサ"    
##  [11] "セトサ"     "セトサ"     "セトサ"     "セトサ"     "セトサ"    
##  [16] "セトサ"     "セトサ"     "セトサ"     "セトサ"     "セトサ"    
##  [21] "セトサ"     "セトサ"     "セトサ"     "セトサ"     "セトサ"    
##  [26] "セトサ"     "セトサ"     "セトサ"     "セトサ"     "セトサ"    
##  [31] "セトサ"     "セトサ"     "セトサ"     "セトサ"     "セトサ"    
##  [36] "セトサ"     "セトサ"     "セトサ"     "セトサ"     "セトサ"    
##  [41] "セトサ"     "セトサ"     "セトサ"     "セトサ"     "セトサ"    
##  [46] "セトサ"     "セトサ"     "セトサ"     "セトサ"     "セトサ"    
##  [51] "versicolor" "versicolor" "versicolor" "versicolor" "versicolor"
##  [56] "versicolor" "versicolor" "versicolor" "versicolor" "versicolor"
##  [61] "versicolor" "versicolor" "versicolor" "versicolor" "versicolor"
##  [66] "versicolor" "versicolor" "versicolor" "versicolor" "versicolor"
##  [71] "versicolor" "versicolor" "versicolor" "versicolor" "versicolor"
##  [76] "versicolor" "versicolor" "versicolor" "versicolor" "versicolor"
##  [81] "versicolor" "versicolor" "versicolor" "versicolor" "versicolor"
##  [86] "versicolor" "versicolor" "versicolor" "versicolor" "versicolor"
##  [91] "versicolor" "versicolor" "versicolor" "versicolor" "versicolor"
##  [96] "versicolor" "versicolor" "versicolor" "versicolor" "versicolor"
## [101] "virginica"  "virginica"  "virginica"  "virginica"  "virginica" 
## [106] "virginica"  "virginica"  "virginica"  "virginica"  "virginica" 
## [111] "virginica"  "virginica"  "virginica"  "virginica"  "virginica" 
## [116] "virginica"  "virginica"  "virginica"  "virginica"  "virginica" 
## [121] "virginica"  "virginica"  "virginica"  "virginica"  "virginica" 
## [126] "virginica"  "virginica"  "virginica"  "virginica"  "virginica" 
## [131] "virginica"  "virginica"  "virginica"  "virginica"  "virginica" 
## [136] "virginica"  "virginica"  "virginica"  "virginica"  "virginica" 
## [141] "virginica"  "virginica"  "virginica"  "virginica"  "virginica" 
## [146] "virginica"  "virginica"  "virginica"  "virginica"  "virginica" 
##    Length     Class      Mode 
##       150 character character 
## versicolor  virginica     セトサ 
##         50         50         50

5.3 欠損データの除外

実験や調査を行った際にデータの一部が欠損しているケースは多々ある。Rでは欠損データはデータフレーム上でNA(Not Available:利用不可能の意味)と表記されている。

関数によっては欠損データを含んでいるデータを受け付けないケースがある。例えばすでに平均値で紹介したようにmean()関数は、そのままでは欠損値を含んだデータは受け付けない。

mean()関数では、関数の引数でna.rm = Tを設定しておくと関数処理の中でNAを除いて処理してくれるが、そもそも与えるデータそのものでNAを除いておくほうが望ましい。

例えば、次のようなデータを処理するケースを考えよう。これはAからFまでの6人に対して10問セットの2種類のクイズを解いてもらった時のスコアである。ただ、1つめのデータセットではDさんのデータが、2つめのデータではEさんのデータが欠損してしまっている。このスコアをもとに、どちらのデータセットが難しいデータだったのかを比べたい。そこで、それぞれの平均や標準偏差を求めてみた。欠損データを含んでいるのでna.rm=Tを設定した。さて、このやり方は正しいだろうか。

d <- data.frame(
  name = c("A", "B", "C", "D", "E", "F"),
  score1 = c(9,8,9,NA,6,7),
  score2 = c(8,7,8,10,NA,7)
)

mean(d$score1, na.rm=T)
## [1] 7.8
sd(d$score1, na.rm=T)
## [1] 1.30384
mean(d$score2, na.rm=T)
## [1] 8
sd(d$score2, na.rm=T)
## [1] 1.224745

このデータの場合、平均や標準偏差を単純比較してはまずい。理由としてscore1ではD以外のデータが使われているのに対して、score2ではE以外のデータが使われているからである。すなわち、それぞれを計算するときに使われたデータが異なっているからである。もしDさんがクイズがすごく得意な人であったならば、Dさんのデータが入っていないscore1の方は、不当に低くくなってしまっているであろうし、逆にEさんがクイズが苦手な人であったならば、Eさんのデータが入っていないscore2の方は不当に高くなっている可能性がある。

データを比較するときには、比較しようとしている項目以外で結果に影響を与えると考えられる項目(条件)は同一条件となっている、ということが大前提である。したがって、上記のようなデータを比較するときには、2種類のクイズのデータが両方ともそろっているA,B,C,Fの4人のデータだけで比較するべきである。

このように、データの一部に欠損値を含んでいるよな観測をすべて除外して、すべての変数でデータがそろっているものだけを残すには、na.omit()関数を用いる。

d.naomit <- na.omit(d)
d.naomit
##   name score1 score2
## 1    A      9      8
## 2    B      8      7
## 3    C      9      8
## 6    F      7      7
mean(d.naomit$score1)
## [1] 8.25
sd(d.naomit$score1)
## [1] 0.9574271
mean(d.naomit$score2)
## [1] 7.5
sd(d.naomit$score2)
## [1] 0.5773503

こうした欠損値は特に任意回答の項目を入れているアンケート調査では間違いなく含まれてくるし、紙でとったアンケート調査においては無回答であったり、択一回答の質問なのに勝手に2つ以上の選ぶなど不正回答をしてくるケースも多々ある。そういう場合にはデータ上はNAとして処理すべきデータとなる。そのような形でNAを含んだデータではデータを読み込んだ時点で上記の通りにna.omit()を実施しておくとよい。

5.4 データの整理方法(ワイド形式とロング形式)

例えば、ある中学校のあるクラス(25人)での1年間の定期テストでの数学の得点をデータとして扱う事例を考えてみよう。

人がそのようなデータを扱うときには以下のような表を作るだろう。

学生番号 前期中間 前期期末 後期中間 後期期末
1 75 94 89 65
2 60 81 70 95
3 68 84 65 70
4 99 78 80 88
5 73 59 85 64
・・・ ・・・ ・・・ ・・・ ・・・
24 85 62 75 88
25 72 77 87 92

このようなデータの並べ方を「ワイド形式」と呼ぶ。ワイド形式は一般に人がみてわかりやすい表形式となる。 それに対して、Rで扱う表は以下のような並べ方のデータを並べている方が扱いやすい。このような並べ方を「ロング形式」と呼ぶ。

データNo. 生徒番号 時期 得点
1 1 前期中間 75
2 1 前期期末 94
3 1 後期中間 89
4 1 後期期末 65
5 2 前期中間 60
6 2 前期期末 81
7 2 後期中間 70
8 2 後期期末 95
9 3 前期中間 68
10 3 前期期末 84
11 3 後期中間 65
12 3 後期期末 70
13 4 前期中間 99
・・・ ・・・ ・・・ ・・・
92 23 後期期末 94
93 24 前期中間 85
94 24 前期期末 62
95 24 後期中間 75
96 24 後期期末 88
97 25 前期中間 72
98 25 前期期末 77
99 25 後期中間 87
100 25 後期期末 92

ワイド形式では列は測定された変数の「属性」を並べるのに対して、ロング形式では「属性」は一つの変数として個別に列立てされて並べる点が異なる。

また、ワイド形式の場合、例えば、数学の得点を並べた表と、国語の得点を並べた表ば別個に作成する必要がある。 一方で、ロング形式の場合、数学の得点の列、国語の得点の列、というように並べて行けばよい。

データ処理という点では、人の目からはわかりづらいがロング形式のデータにしておく習慣を持っておこう。

なお、Rではワイド形式のデータをロング形式のデータに変換する関数が

6 基本的な検定

6.1 平均と分散の比較

T検定とF検定 結果の可視化 結果の報告の仕方

6.2 比率の比較

X2乗検定 Fisherの正確確率検定 結果の報告の仕方

6.3 多重比較

検定の多重性問題 結果の可視化 結果の報告の仕方

7 分散分析

一要因分散分析 結果の報告の仕方 二要因分散分析 結果の報告の仕方 調整変数を含んだ分散分析

対応のあるデータでの分散分析 分散分析のタイプ

8 関係性の分析

8.1 相関分析

8.1.1 共分散と相関

8.1.2 疑似相関と重相関

8.2 回帰分析・重回帰分析

8.2.1 回帰直線の可視化

9 関係性の分析(2)

9.1 一般化線形モデル

9.2 混合モデル

10 共分散構造分析

10.1 パス解析

10.2 構造方程式モデリング

11 主成分分析と因子分析

12 クラスター分析

13 テキストマイニング

14 R・RStudioのさらなる活用

14.1 プログラムを活用した分析と自作関数の利用

14.2 RmarkdownとHTML出力

14.3 GitHubとPosit Cloudとの連携